更新日期: 2022/06/01 来源: https://gitee.com/weharmony/kernel_liteos_a_note
los_membox.c
浏览该文件的文档.
1/*!
2 * @file los_membox.c
3 * @brief 静态内存池主文件
4 * @link
5 @verbatim
6
7 使用场景
8 当用户需要使用固定长度的内存时,可以通过静态内存分配的方式获取内存,一旦使用完毕,
9 通过静态内存释放函数归还所占用内存,使之可以重复使用。
10
11 开发流程
12 通过make menuconfig配置静态内存管理模块。
13 规划一片内存区域作为静态内存池。
14 调用LOS_MemboxInit初始化静态内存池。
15 初始化会将入参指定的内存区域分割为N块(N值取决于静态内存总大小和块大小),将所有内存块挂到空闲链表,在内存起始处放置控制头。
16 调用LOS_MemboxAlloc接口分配静态内存。
17 系统将会从空闲链表中获取第一个空闲块,并返回该内存块的起始地址。
18 调用LOS_MemboxClr接口。将入参地址对应的内存块清零。
19 调用LOS_MemboxFree接口。将该内存块加入空闲链表。
20
21 注意事项
22 静态内存池区域,如果是通过动态内存分配方式获得的,在不需要静态内存池时,
23 需要释放该段内存,避免发生内存泄露。
24 静态内存不常用,因为需要使用者去确保不会超出使用范围
25
26 @endverbatim
27 * @version
28 * @author weharmonyos.com | 鸿蒙研究站 | 每天死磕一点点
29 * @date 2022-04-02
30 */
31
32/*
33 * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
34 * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
35 *
36 * Redistribution and use in source and binary forms, with or without modification,
37 * are permitted provided that the following conditions are met:
38 *
39 * 1. Redistributions of source code must retain the above copyright notice, this list of
40 * conditions and the following disclaimer.
41 *
42 * 2. Redistributions in binary form must reproduce the above copyright notice, this list
43 * of conditions and the following disclaimer in the documentation and/or other materials
44 * provided with the distribution.
45 *
46 * 3. Neither the name of the copyright holder nor the names of its contributors may be used
47 * to endorse or promote products derived from this software without specific prior written
48 * permission.
49 *
50 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
51 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
52 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
53 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
54 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
55 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
56 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
57 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
58 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
59 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
60 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
61 */
62
63#include "los_membox.h"
64#include "los_hwi.h"
65#include "los_spinlock.h"
66
67#ifdef LOSCFG_AARCH64
68#define OS_MEMBOX_MAGIC 0xa55a5aa5a55a5aa5 //魔法数字,@note_good 点赞,设计的很精巧,node内容从下一个节点地址变成魔法数字
69#else
70#define OS_MEMBOX_MAGIC 0xa55a5aa5
71#endif
72#define OS_MEMBOX_SET_MAGIC(addr) \
73 ((LOS_MEMBOX_NODE *)(addr))->pstNext = (LOS_MEMBOX_NODE *)OS_MEMBOX_MAGIC //设置魔法数字
74#define OS_MEMBOX_CHECK_MAGIC(addr) \
75 ((((LOS_MEMBOX_NODE *)(addr))->pstNext == (LOS_MEMBOX_NODE *)OS_MEMBOX_MAGIC) ? LOS_OK : LOS_NOK)
76
77#define OS_MEMBOX_USER_ADDR(addr) \
78 ((VOID *)((UINT8 *)(addr) + OS_MEMBOX_NODE_HEAD_SIZE)) //@note_good 使用之前要去掉节点信息,太赞了! 很艺术化!!
79#define OS_MEMBOX_NODE_ADDR(addr) \
80 ((LOS_MEMBOX_NODE *)(VOID *)((UINT8 *)(addr) - OS_MEMBOX_NODE_HEAD_SIZE)) //节块 = (节头 + 节体) addr = 节体
81/* spinlock for mem module, only available on SMP mode */
82LITE_OS_SEC_BSS SPIN_LOCK_INIT(g_memboxSpin);
83#define MEMBOX_LOCK(state) LOS_SpinLockSave(&g_memboxSpin, &(state)) ///< 获取静态内存池自旋锁
84#define MEMBOX_UNLOCK(state) LOS_SpinUnlockRestore(&g_memboxSpin, (state))///< 释放静态内存池自旋锁
85/// 检查静态内存块
86STATIC INLINE UINT32 OsCheckBoxMem(const LOS_MEMBOX_INFO *boxInfo, const VOID *node)
87{
88 UINT32 offset;
89
90 if (boxInfo->uwBlkSize == 0) {
91 return LOS_NOK;
92 }
93
94 offset = (UINT32)((UINTPTR)node - (UINTPTR)(boxInfo + 1));
95 if ((offset % boxInfo->uwBlkSize) != 0) {
96 return LOS_NOK;
97 }
98
99 if ((offset / boxInfo->uwBlkSize) >= boxInfo->uwBlkNum) {
100 return LOS_NOK;
101 }
102
103 return OS_MEMBOX_CHECK_MAGIC(node);//检查魔法数字是否被修改过了
104}
105/// 初始化一个静态内存池,根据入参设定其起始地址、总大小及每个内存块大小
106LITE_OS_SEC_TEXT_INIT UINT32 LOS_MemboxInit(VOID *pool, UINT32 poolSize, UINT32 blkSize)
107{
108 LOS_MEMBOX_INFO *boxInfo = (LOS_MEMBOX_INFO *)pool;//在内存起始处安置池头
109 LOS_MEMBOX_NODE *node = NULL;
110 UINT32 index;
111 UINT32 intSave;
112
113 if (pool == NULL) {
114 return LOS_NOK;
115 }
116
117 if (blkSize == 0) {
118 return LOS_NOK;
119 }
120
121 if (poolSize < sizeof(LOS_MEMBOX_INFO)) {
122 return LOS_NOK;
123 }
124
125 MEMBOX_LOCK(intSave);
126 boxInfo->uwBlkSize = LOS_MEMBOX_ALIGNED(blkSize + OS_MEMBOX_NODE_HEAD_SIZE); //节块总大小(节头+节体)
127 boxInfo->uwBlkNum = (poolSize - sizeof(LOS_MEMBOX_INFO)) / boxInfo->uwBlkSize;//总节块数量
128 boxInfo->uwBlkCnt = 0; //已分配的数量
129 if (boxInfo->uwBlkNum == 0) {//只有0块的情况
130 MEMBOX_UNLOCK(intSave);
131 return LOS_NOK;
132 }
133
134 node = (LOS_MEMBOX_NODE *)(boxInfo + 1);//去除池头,找到第一个节块位置
135
136 boxInfo->stFreeList.pstNext = node;//池头空闲链表指向第一个节块
137
138 for (index = 0; index < boxInfo->uwBlkNum - 1; ++index) {//切割节块,挂入空闲链表
139 node->pstNext = OS_MEMBOX_NEXT(node, boxInfo->uwBlkSize);//按块大小切割好,统一由pstNext指向
140 node = node->pstNext;//node存储了下一个节点的地址信息
141 }
142
143 node->pstNext = NULL;//最后一个为null
144
145 MEMBOX_UNLOCK(intSave);
146
147 return LOS_OK;
148}
149/// 从指定的静态内存池中申请一块静态内存块,整个内核源码只有 OsSwtmrScan中用到了静态内存.
150LITE_OS_SEC_TEXT VOID *LOS_MemboxAlloc(VOID *pool)
151{
152 LOS_MEMBOX_INFO *boxInfo = (LOS_MEMBOX_INFO *)pool;
153 LOS_MEMBOX_NODE *node = NULL;
154 LOS_MEMBOX_NODE *nodeTmp = NULL;
155 UINT32 intSave;
156
157 if (pool == NULL) {
158 return NULL;
159 }
160
161 MEMBOX_LOCK(intSave);
162 node = &(boxInfo->stFreeList);//拿到空闲单链表
163 if (node->pstNext != NULL) {//不需要遍历链表,因为这是空闲链表
164 nodeTmp = node->pstNext;//先记录要使用的节点
165 node->pstNext = nodeTmp->pstNext;//不再空闲了,把节点摘出去了.
166 OS_MEMBOX_SET_MAGIC(nodeTmp);//为已使用的节块设置魔法数字
167 boxInfo->uwBlkCnt++;//已使用块数增加
168 }
169 MEMBOX_UNLOCK(intSave);
170
171 return (nodeTmp == NULL) ? NULL : OS_MEMBOX_USER_ADDR(nodeTmp);//返回可用的虚拟地址
172}
173/// 释放指定的一块静态内存块
174LITE_OS_SEC_TEXT UINT32 LOS_MemboxFree(VOID *pool, VOID *box)
175{
176 LOS_MEMBOX_INFO *boxInfo = (LOS_MEMBOX_INFO *)pool;
177 UINT32 ret = LOS_NOK;
178 UINT32 intSave;
179
180 if ((pool == NULL) || (box == NULL)) {
181 return LOS_NOK;
182 }
183
184 MEMBOX_LOCK(intSave);
185 do {
186 LOS_MEMBOX_NODE *node = OS_MEMBOX_NODE_ADDR(box);//通过节体获取节块首地址
187 if (OsCheckBoxMem(boxInfo, node) != LOS_OK) {
188 break;
189 }
190
191 node->pstNext = boxInfo->stFreeList.pstNext;//节块指向空闲链表表头
192 boxInfo->stFreeList.pstNext = node;//空闲链表表头反指向它,意味节块排到第一,下次申请将首个分配它
193 boxInfo->uwBlkCnt--;//已经使用的内存块减一
194 ret = LOS_OK;
195 } while (0);//将被编译时优化
196 MEMBOX_UNLOCK(intSave);
197
198 return ret;
199}
200/// 清零指定静态内存块的内容
201LITE_OS_SEC_TEXT_MINOR VOID LOS_MemboxClr(VOID *pool, VOID *box)
202{
203 LOS_MEMBOX_INFO *boxInfo = (LOS_MEMBOX_INFO *)pool;
204
205 if ((pool == NULL) || (box == NULL)) {
206 return;
207 }
208 //将魔法数字一并清除了.
209 (VOID)memset_s(box, (boxInfo->uwBlkSize - OS_MEMBOX_NODE_HEAD_SIZE), 0,
210 (boxInfo->uwBlkSize - OS_MEMBOX_NODE_HEAD_SIZE));
211}
212/// 打印指定静态内存池所有节点信息(打印等级是LOS_INFO_LEVEL),包括内存池起始地址、
213/// 内存块大小、总内存块数量、每个空闲内存块的起始地址、所有内存块的起始地址
214LITE_OS_SEC_TEXT_MINOR VOID LOS_ShowBox(VOID *pool)
215{
216 UINT32 index;
217 UINT32 intSave;
218 LOS_MEMBOX_INFO *boxInfo = (LOS_MEMBOX_INFO *)pool;
219 LOS_MEMBOX_NODE *node = NULL;
220
221 if (pool == NULL) {
222 return;
223 }
224 MEMBOX_LOCK(intSave);
225 PRINT_INFO("membox(%p,0x%x,0x%x):\r\n", pool, boxInfo->uwBlkSize, boxInfo->uwBlkNum);
226 PRINT_INFO("free node list:\r\n");
227
228 for (node = boxInfo->stFreeList.pstNext, index = 0; node != NULL;
229 node = node->pstNext, ++index) {
230 PRINT_INFO("(%u,%p)\r\n", index, node);
231 }
232
233 PRINT_INFO("all node list:\r\n");
234 node = (LOS_MEMBOX_NODE *)(boxInfo + 1);
235 for (index = 0; index < boxInfo->uwBlkNum; ++index, node = OS_MEMBOX_NEXT(node, boxInfo->uwBlkSize)) {
236 PRINT_INFO("(%u,%p,%p)\r\n", index, node, node->pstNext);
237 }
238 MEMBOX_UNLOCK(intSave);
239}
240/// 获取指定静态内存池的信息,包括内存池中总内存块数量、已经分配出去的内存块数量、每个内存块的大小
241LITE_OS_SEC_TEXT_MINOR UINT32 LOS_MemboxStatisticsGet(const VOID *boxMem, UINT32 *maxBlk,
242 UINT32 *blkCnt, UINT32 *blkSize)
243{
244 if ((boxMem == NULL) || (maxBlk == NULL) || (blkCnt == NULL) || (blkSize == NULL)) {
245 return LOS_NOK;
246 }
247
248 *maxBlk = ((OS_MEMBOX_S *)boxMem)->uwBlkNum;
249 *blkCnt = ((OS_MEMBOX_S *)boxMem)->uwBlkCnt;
250 *blkSize = ((OS_MEMBOX_S *)boxMem)->uwBlkSize;
251
252 return LOS_OK;
253}
254
LITE_OS_SEC_TEXT_MINOR VOID LOS_MemboxClr(VOID *pool, VOID *box)
清零指定静态内存块的内容
Definition: los_membox.c:201
LITE_OS_SEC_TEXT VOID * LOS_MemboxAlloc(VOID *pool)
从指定的静态内存池中申请一块静态内存块,整个内核源码只有 OsSwtmrScan中用到了静态内存.
Definition: los_membox.c:150
LITE_OS_SEC_TEXT_MINOR VOID LOS_ShowBox(VOID *pool)
show membox info.
Definition: los_membox.c:214
LITE_OS_SEC_TEXT UINT32 LOS_MemboxFree(VOID *pool, VOID *box)
释放指定的一块静态内存块
Definition: los_membox.c:174
LITE_OS_SEC_TEXT_INIT UINT32 LOS_MemboxInit(VOID *pool, UINT32 poolSize, UINT32 blkSize)
初始化一个静态内存池,根据入参设定其起始地址、总大小及每个内存块大小
Definition: los_membox.c:106
LITE_OS_SEC_TEXT_MINOR UINT32 LOS_MemboxStatisticsGet(const VOID *boxMem, UINT32 *maxBlk, UINT32 *blkCnt, UINT32 *blkSize)
获取指定静态内存池的信息,包括内存池中总内存块数量、已经分配出去的内存块数量、每个内存块的大小
Definition: los_membox.c:241
STATIC INLINE UINT32 OsCheckBoxMem(const LOS_MEMBOX_INFO *boxInfo, const VOID *node)
检查静态内存块
Definition: los_membox.c:86
LITE_OS_SEC_BSS SPIN_LOCK_INIT(g_memboxSpin)
unsigned long UINTPTR
Definition: los_typedef.h:68
unsigned int UINT32
Definition: los_typedef.h:57
UINT32 uwBlkSize
Definition: los_membox.h:65
UINT32 uwBlkNum
Definition: los_membox.h:66
UINT32 uwBlkCnt
Definition: los_membox.h:67
LOS_MEMBOX_NODE stFreeList
Definition: los_membox.h:68
struct tagMEMBOX_NODE * pstNext
Definition: los_membox.h:57