更新日期: 2022/06/01 来源: https://gitee.com/weharmony/kernel_liteos_a_note
los_asid.c
浏览该文件的文档.
1/*!
2 * @file los_asid.c
3 * @brief
4 * @link http://weharmonyos.com/blog/14.html
5 * @verbatim
6 asid(Adress Space ID) 进程标识符,属于CP15协处理器的C13号寄存器,ASID可用来唯一标识进程,并为进程提供地址空间保护。
7 当TLB试图解析虚拟页号时,它确保当前运行进程的ASID与虚拟页相关的ASID相匹配。如果不匹配,那么就作为TLB失效。
8 除了提供地址空间保护外,ASID允许TLB同时包含多个进程的条目。如果TLB不支持独立的ASID,每次选择一个页表时(例如,上下文切换时),
9 TLB就必须被冲刷(flushed)或删除,以确保下一个进程不会使用错误的地址转换。
10
11 TLB页表中有一个bit来指明当前的entry是global(nG=0,所有process都可以访问)还是non-global(nG=1,only本process允许访问)。
12 如果是global类型,则TLB中不会tag ASID;如果是non-global类型,则TLB会tag上ASID,
13 且MMU在TLB中查询时需要判断这个ASID和当前进程的ASID是否一致,只有一致才证明这条entry当前process有权限访问。
14
15 看到了吗?如果每次mmu上下文切换时,把TLB全部刷新已保证TLB中全是新进程的映射表,固然是可以,但效率太低了!!!
16 进程的切换其实是秒级亚秒级的,地址的虚实转换是何等的频繁啊,怎么会这么现实呢,
17 真实的情况是TLB中有很多很多其他进程占用的物理内存的记录还在,当然他们对物理内存的使用权也还在。
18 所以当应用程序 new了10M内存以为是属于自己的时候,其实在内核层面根本就不属于你,还是别人在用,
19 只有你用了1M的那一瞬间真正1M物理内存才属于你,而且当你的进程被其他进程切换后,很大可能你用的那1M也已经不在物理内存中了,
20 已经被置换到硬盘上了。明白了吗?只关注应用开发的同学当然可以说这关我鸟事,给我的感觉有就行了,但想熟悉内核的同学就必须要明白,
21 这是每分每秒都在发生的事情。
22 * @endverbatim
23 * @version
24 * @author weharmonyos.com | 鸿蒙研究站 | 每天死磕一点点
25 * @date 2021-11-16
26 *
27 * @history
28 *
29 */
30/*
31 * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
32 * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
33 *
34 * Redistribution and use in source and binary forms, with or without modification,
35 * are permitted provided that the following conditions are met:
36 *
37 * 1. Redistributions of source code must retain the above copyright notice, this list of
38 * conditions and the following disclaimer.
39 *
40 * 2. Redistributions in binary form must reproduce the above copyright notice, this list
41 * of conditions and the following disclaimer in the documentation and/or other materials
42 * provided with the distribution.
43 *
44 * 3. Neither the name of the copyright holder nor the names of its contributors may be used
45 * to endorse or promote products derived from this software without specific prior written
46 * permission.
47 *
48 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
49 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
50 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
51 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
52 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
53 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
54 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
55 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
56 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
57 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
58 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
59 */
60
61/**
62 * @defgroup los_asid mmu address space id
63 * @ingroup kernel
64 */
65
66#include "los_asid.h"
67#include "los_bitmap.h"
68#include "los_spinlock.h"
70
71
72#ifdef LOSCFG_KERNEL_VM
73
74STATIC SPIN_LOCK_INIT(g_cpuAsidLock); ///< asid专属自旋锁
75STATIC UINTPTR g_asidPool[BITMAP_NUM_WORDS(1UL << MMU_ARM_ASID_BITS)]; ///< 地址空间ID池 , 2^8 = 256 个
76
77/* allocate and free asid | 分配地址空间ID */
79{
80 UINT32 flags;
81 LOS_SpinLockSave(&g_cpuAsidLock, &flags);
82 UINT32 firstZeroBit = LOS_BitmapFfz(g_asidPool, 1UL << MMU_ARM_ASID_BITS);//找到第一个0位,即可分配位
83 if (firstZeroBit >= 0 && firstZeroBit < (1UL << MMU_ARM_ASID_BITS)) {
84 LOS_BitmapSetNBits(g_asidPool, firstZeroBit, 1);//设为已分配
85 *asid = firstZeroBit;//由参数带走
86 LOS_SpinUnlockRestore(&g_cpuAsidLock, flags);
87 return LOS_OK;
88 }
89
90 LOS_SpinUnlockRestore(&g_cpuAsidLock, flags);
91 return firstZeroBit;
92}
93
94
95/// 释放 asid
97{
98 UINT32 flags;
99 LOS_SpinLockSave(&g_cpuAsidLock, &flags);
100 LOS_BitmapClrNBits(g_asidPool, asid, 1);//清空对应位,可继续分配给其他进程
101 LOS_SpinUnlockRestore(&g_cpuAsidLock, flags);
102}
103#endif
104
VOID LOS_BitmapClrNBits(UINTPTR *bitmap, UINT32 start, UINT32 numsClear)
从start位置开始 清除numsSet个bit位置0 ,对状态字的连续标志位进行清0操作
Definition: los_bitmap.c:126
INT32 LOS_BitmapFfz(UINTPTR *bitmap, UINT32 numBits)
从numBits位置开始找到第一个0位
Definition: los_bitmap.c:146
VOID LOS_BitmapSetNBits(UINTPTR *bitmap, UINT32 start, UINT32 numsSet)
从start位置开始设置numsSet个bit位 置1
Definition: los_bitmap.c:106
STATIC SPIN_LOCK_INIT(g_cpuAsidLock)
asid专属自旋锁
STATIC UINTPTR g_asidPool[BITMAP_NUM_WORDS(1UL<< MMU_ARM_ASID_BITS)]
地址空间ID池 , 2^8 = 256 个
Definition: los_asid.c:75
status_t OsAllocAsid(UINT32 *asid)
Definition: los_asid.c:78
VOID OsFreeAsid(UINT32 asid)
释放 asid
Definition: los_asid.c:96
VOID LOS_SpinUnlockRestore(SPIN_LOCK_S *lock, UINT32 intSave)
Definition: los_spinlock.c:108
VOID LOS_SpinLockSave(SPIN_LOCK_S *lock, UINT32 *intSave)
Definition: los_spinlock.c:98
int status_t
Definition: los_typedef.h:205
unsigned long UINTPTR
Definition: los_typedef.h:68
unsigned int UINT32
Definition: los_typedef.h:57