123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- /*
- * Copyright (c) 2006-2018, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2021-05-06 Jesven first version
- */
- #include <rtthread.h>
- #include <rthw.h>
- #ifdef RT_USING_USERSPACE
- #include <mmu.h>
- #include <lwp_mm_area.h>
- #endif
- #include <ioremap.h>
- #ifdef RT_USING_USERSPACE
- static struct lwp_avl_struct *k_map_area;
- extern rt_mmu_info mmu_info;
- static void _iounmap_range(void *addr, size_t size)
- {
- void *va = RT_NULL, *pa = RT_NULL;
- int i = 0;
- for (va = addr, i = 0; i < size; va = (void *)((char *)va + ARCH_PAGE_SIZE), i += ARCH_PAGE_SIZE)
- {
- pa = rt_hw_mmu_v2p(&mmu_info, va);
- if (pa)
- {
- rt_hw_mmu_unmap(&mmu_info, va, ARCH_PAGE_SIZE);
- }
- }
- }
- static void *_ioremap_type(void *p_addr, size_t size, int type)
- {
- rt_base_t level;
- void *v_addr = NULL;
- size_t attr;
- switch (type)
- {
- case MM_AREA_TYPE_PHY:
- attr = MMU_MAP_K_DEVICE;
- break;
- case MM_AREA_TYPE_PHY_CACHED:
- attr = MMU_MAP_K_RWCB;
- break;
- default:
- return v_addr;
- }
- level = rt_hw_interrupt_disable();
- v_addr = rt_hw_mmu_map(&mmu_info, 0, p_addr, size, attr);
- if (v_addr)
- {
- int ret = lwp_map_area_insert(&k_map_area, (size_t)v_addr, size, type);
- if (ret != 0)
- {
- _iounmap_range(v_addr, size);
- v_addr = NULL;
- }
- }
- rt_hw_interrupt_enable(level);
- return v_addr;
- }
- void *rt_ioremap(void *p_addr, size_t size)
- {
- return _ioremap_type(p_addr, size, MM_AREA_TYPE_PHY);
- }
- void *rt_ioremap_nocache(void *p_addr, size_t size)
- {
- return _ioremap_type(p_addr, size, MM_AREA_TYPE_PHY);
- }
- void *rt_ioremap_cached(void *p_addr, size_t size)
- {
- return _ioremap_type(p_addr, size, MM_AREA_TYPE_PHY_CACHED);
- }
- #else
- void *rt_ioremap(void *p_addr, size_t size)
- {
- return p_addr;
- }
- void *rt_ioremap_nocache(void *paddr, size_t size)
- {
- return p_addr;
- }
- void *rt_ioremap_cached(void *paddr, size_t size)
- {
- return p_addr;
- }
- #endif
- void rt_iounmap(volatile void *vaddr)
- {
- #ifdef RT_USING_USERSPACE
- rt_base_t level;
- struct lwp_avl_struct *ma_avl_node;
- level = rt_hw_interrupt_disable();
- ma_avl_node = lwp_map_find(k_map_area, (size_t)vaddr);
- if (ma_avl_node)
- {
- struct rt_mm_area_struct *ma = (struct rt_mm_area_struct *)ma_avl_node->data;
- _iounmap_range((void *)ma->addr, ma->size);
- lwp_map_area_remove(&k_map_area, (size_t)vaddr);
- }
- rt_hw_interrupt_enable(level);
- #endif
- }
|