/* * Copyright (c) 2006-2020, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2019-10-28 Jesven first version * 2021-02-12 lizhirui add 64-bit support for lwp_brk * 2023-09-19 Shell add lwp_user_memory_remap_to_kernel */ #ifndef __LWP_USER_MM_H__ #define __LWP_USER_MM_H__ #include #include #ifdef ARCH_MM_MMU #include #include #include #include #include #ifdef __cplusplus extern "C" { #endif #define LWP_MAP_FLAG_NONE 0x0000 #define LWP_MAP_FLAG_NOCACHE 0x0001 #define LWP_MAP_FLAG_MAP_FIXED 0x00010000ul #define LWP_MAP_FLAG_PREFETCH 0x00020000ul /** * @brief Map files or devices into memory * It will create a new mapping in the virtual address space of the target lwp * * @param lwp target process * @param addr address from user space * @param length length in bytes of mapping * @param prot protect attribution of mapping * @param flags flags of control * @param fd file descriptor * @param pgoffset offset to fd in 4096 bytes unit * @return void* the address is successful, otherwise return MAP_FAILED */ void* lwp_mmap2(struct rt_lwp *lwp, void *addr, size_t length, int prot, int flags, int fd, off_t pgoffset); /** * @brief Unmap memory region in user space * * @param lwp target process * @param addr address to unmap * @param length length in bytes of unmapping * @return int errno */ int lwp_munmap(struct rt_lwp *lwp, void *addr, size_t length); void *lwp_mremap(struct rt_lwp *lwp, void *old_address, size_t old_size, size_t new_size, int flags, void *new_address); /** * @brief Test if address from user is accessible address by user * * @param lwp target process * @param addr address from user space * @param size the bytes to access * @return int RT_FALSE/RT_TRUE */ int lwp_user_accessible_ext(struct rt_lwp *lwp, void *addr, size_t size); /** * @brief Test if address from user is accessible address by user * Same as lwp_user_accessible_ext except that lwp is current lwp * * @param addr address from user space * @param size the bytes to access * @return int RT_FALSE/RT_TRUE */ int lwp_user_accessable(void *addr, size_t size); /** * @brief Copy n bytes data from src to dst. * Same as std libc memcpy, except that both src and dst may come from * user space. lwp_memcpy will test and select the implementation based * on the memory attribution on run-time * * @param dst where the data writes to * @param src where the data comes from * @param size the bytes to copy * @return void* the destination address */ void *lwp_memcpy(void * __restrict dst, const void * __restrict src, size_t size); /** * @brief memcpy from address in user address space to kernel space buffer * * @param lwp target process * @param dst kernel space address where the data writes to * @param src user space address where the data comes from * @param size the bytes to copy * @return size_t the bytes copied */ size_t lwp_data_get(struct rt_lwp *lwp, void *dst, void *src, size_t size); /** * @brief lwp_data_get except that lwp is current lwp * * @param dst kernel space address where the data writes to * @param src user space address where the data comes from * @param size the bytes to copy * @return size_t the bytes copied */ size_t lwp_get_from_user(void *dst, void *src, size_t size); /** * @brief memcpy from kernel space buffer to address in user address space * * @param lwp target process * @param dst user space address where the data writes to * @param src kernel space address where the data comes from * @param size the bytes to copy * @return size_t the bytes copied */ size_t lwp_data_put(struct rt_lwp *lwp, void *dst, void *src, size_t size); /** * @brief lwp_data_put except that lwp is current lwp * * @param dst user space address where the data writes to * @param src kernel space address where the data comes from * @param size the bytes to copy * @return size_t the bytes copied */ size_t lwp_put_to_user(void *dst, void *src, size_t size); /** * @brief memset to address in user address space * * @param lwp target process * @param dst user space address where the data writes to * @param c the value to write * @param size the bytes to copy * @return size_t the bytes written */ size_t lwp_data_set(struct rt_lwp *lwp, void *dst, int c, size_t size); int lwp_user_space_init(struct rt_lwp *lwp, rt_bool_t is_fork); void lwp_unmap_user_space(struct rt_lwp *lwp); int lwp_unmap_user(struct rt_lwp *lwp, void *va); void *lwp_map_user(struct rt_lwp *lwp, void *map_va, size_t map_size, rt_bool_t text); rt_varea_t lwp_map_user_varea(struct rt_lwp *lwp, void *map_va, size_t map_size); /* check LWP_MAP_FLAG_* */ rt_varea_t lwp_map_user_varea_ext(struct rt_lwp *lwp, void *map_va, size_t map_size, size_t flags); void *lwp_map_user_phy(struct rt_lwp *lwp, void *map_va, void *map_pa, size_t map_size, rt_bool_t cached); int lwp_unmap_user_phy(struct rt_lwp *lwp, void *va); rt_base_t lwp_brk(void *addr); size_t lwp_user_strlen(const char *s); size_t lwp_user_strlen_ext(struct rt_lwp *lwp, const char *s); size_t lwp_strlen(struct rt_lwp *lwp, const char *s); int lwp_fork_aspace(struct rt_lwp *dest_lwp, struct rt_lwp *src_lwp); void lwp_data_cache_flush(struct rt_lwp *lwp, void *vaddr, size_t size); static inline void *_lwp_v2p(struct rt_lwp *lwp, void *vaddr) { return rt_hw_mmu_v2p(lwp->aspace, vaddr); } static inline void *lwp_v2p(struct rt_lwp *lwp, void *vaddr) { RD_LOCK(lwp->aspace); void *paddr = _lwp_v2p(lwp, vaddr); RD_UNLOCK(lwp->aspace); return paddr; } /** * @brief Remapping user space memory region to kernel * * @warning the remapped region in kernel should be unmapped after usage * * @param lwp target process * @param uaddr user space address where the data writes to * @param length the bytes to redirect * @return void * the redirection address in kernel space */ void *lwp_user_memory_remap_to_kernel(rt_lwp_t lwp, void *uaddr, size_t length); rt_inline rt_size_t lwp_user_mm_flag_to_kernel(int flags) { rt_size_t k_flags = 0; if (flags & MAP_FIXED) k_flags |= MMF_MAP_FIXED; if (flags & (MAP_PRIVATE | MAP_ANON | MAP_ANONYMOUS)) k_flags |= MMF_MAP_PRIVATE; if (flags & MAP_SHARED) k_flags |= MMF_MAP_SHARED; return k_flags; } #ifndef MMU_MAP_U_ROCB #define MMU_MAP_U_ROCB MMU_MAP_U_RWCB #endif /* MMU_MAP_U_ROCB */ rt_inline rt_size_t lwp_user_mm_attr_to_kernel(int prot) { RT_UNUSED(prot); rt_size_t k_attr = 0; #ifdef LWP_USING_MPROTECT if ((prot & PROT_EXEC) || (prot & PROT_WRITE) || ((prot & PROT_READ) && (prot & PROT_WRITE))) k_attr = MMU_MAP_U_RWCB; else if (prot == PROT_NONE) k_attr = MMU_MAP_K_RWCB; else k_attr = MMU_MAP_U_ROCB; #else /* !LWP_USING_MPROTECT */ k_attr = MMU_MAP_U_RWCB; #endif /* LWP_USING_MPROTECT */ return k_attr; } #ifdef __cplusplus } #endif #endif #endif /*__LWP_USER_MM_H__*/