XWOS API  4.0
XWOS C/C++ API参考手册
载入中...
搜索中...
未找到
伙伴算法内存块分配器
伙伴算法内存块分配器 的协作图:

结构体

struct  xwmm_bma_orderlist
 阶链表 更多...
 
struct  xwmm_bma_bcb
 块控制块 更多...
 
struct  xwmm_bma
 伙伴算法内存块分配器 更多...
 

宏定义

#define XWMM_BMA_MAX_ORDER   ((xwu8_t)126)
 
#define XWMM_BMA_COMBINED   ((xwu8_t)127)
 
#define XWMM_BMA_ORDER_MASK   ((xwu8_t)0x7F)
 
#define XWMM_BMA_INUSED   ((xwu8_t)0x80)
 
#define XWMM_BMA_RAWOBJ_DEF(name, blkodr)
 定义伙伴算法内存块分配器结构体的RAW内存空间, 用于初始化伙伴算法内存块分配器结构体
 

函数

xwer_t xwmm_bma_init (struct xwmm_bma *bma, const char *name, xwptr_t origin, xwsz_t size, xwsz_t blksize, xwsz_t blkodr)
 XWMM API:初始化伙伴算法内存块分配器
 
xwer_t xwmm_bma_alloc (struct xwmm_bma *bma, xwsq_t order, void **membuf)
 XWMM API:申请一块连续的内存
 
xwer_t xwmm_bma_free (struct xwmm_bma *bma, void *mem)
 XWMM API:释放内存块
 

详细描述

伙伴算法分配器在分配时,会将内存不断地二等分,直到切割到能满足内存的最小尺寸为止。 释放时会检查与之相邻并等长的内存块(称为伙伴)是否空闲, 如果是,就和“伙伴”合并成更大的内存块, 然后继续检测合并后的内存块是否也存在空闲的伙伴,一直向上合并到不能合并为止。

宏定义说明

◆ XWMM_BMA_COMBINED

#define XWMM_BMA_COMBINED   ((xwu8_t)127)

块已被合并

在文件 bma.h45 行定义.

◆ XWMM_BMA_INUSED

#define XWMM_BMA_INUSED   ((xwu8_t)0x80)

块正在被使用

在文件 bma.h47 行定义.

◆ XWMM_BMA_MAX_ORDER

#define XWMM_BMA_MAX_ORDER   ((xwu8_t)126)

最大的阶

在文件 bma.h44 行定义.

◆ XWMM_BMA_ORDER_MASK

#define XWMM_BMA_ORDER_MASK   ((xwu8_t)0x7F)

阶的掩码

在文件 bma.h46 行定义.

◆ XWMM_BMA_RAWOBJ_DEF

#define XWMM_BMA_RAWOBJ_DEF (   name,
  blkodr 
)
值:
xwu8_t name[sizeof(struct xwmm_bma) + \
sizeof(struct xwmm_bma_orderlist[(blkodr) + 1U]) + \
sizeof(struct xwmm_bma_bcb[1U << (blkodr)])]
uint8_t xwu8_t
Definition type.h:194
块控制块
Definition bma.h:70
阶链表
Definition bma.h:63
伙伴算法内存块分配器
Definition bma.h:80
xwsq_t blkodr
Definition bma.h:84

定义伙伴算法内存块分配器结构体的RAW内存空间, 用于初始化伙伴算法内存块分配器结构体

参数
[in]name内存数组名
[in]blkodr伙伴算法内存块分配器中单位内存块的数量,以2的blkodr次方形式表示

在文件 bma.h55 行定义.

函数说明

◆ xwmm_bma_alloc()

xwer_t xwmm_bma_alloc ( struct xwmm_bma bma,
xwsq_t  order,
void **  membuf 
)

XWMM API:申请一块连续的内存

参数
[in]bma伙伴算法内存块分配器对象的指针
[in]order块数量的阶,内存块大小: ((1 << order) * bma->blksize)
[out]membuf指向地址缓存的指针,通过此指针缓存返回申请到的内存的首地址
返回
错误码
返回值
-EFAULT空指针
-ERANGEorder无效
-ENOMEM内存不足
注解
  • 同步/异步:同步
  • 上下文:中断、中断底半部、线程
  • 重入性:可重入

<No error

在文件 bma.c311 行定义.

312{
313 xwer_t rc;
314 xwreg_t flag;
315 xwsq_t o;
316 struct xwmm_bma_orderlist * ol;
317 struct xwmm_bma_bcb * bcb;
318
319 XWOS_VALIDATE((bma), "nullptr", -EFAULT);
320 XWOS_VALIDATE((membuf), "nullptr", -EFAULT);
321 XWOS_VALIDATE((order <= (xwsq_t)XWSSQ_MAX), "out-of-range", -ERANGE);
322
323 ol = NULL;
324 bcb = err_ptr(-ENOENT);
325 xwos_splk_lock_cpuirqsv(&bma->lock, &flag);
326 for (o = order; o <= bma->blkodr; o++) {
327 ol = &bma->orderlists[o];
328 bcb = xwmm_bma_orderlist_choose(bma, ol);
329 if (!is_err(bcb)) {
330 break;
331 }
332 }
333 if (is_err(bcb)) { // cppcheck-suppress [misra-c2012-14.4]
334 xwos_splk_unlock_cpuirqrs(&bma->lock, flag);
335 rc = -ENOMEM;
336 *membuf = NULL;
337 } else {
338 xwmm_bmalogf(DEBUG,
339 "[ALLOC] bcb(idx:0x%lX,odr:0x%X)\n",
340 (((xwptr_t)bcb - (xwptr_t)xwmm_bma->bcbs) /
341 sizeof(struct xwmm_bma_bcb)),
342 bcb->order);
343 xwmm_bma_divide_block(bma, bcb, order, ol);
344 xwos_splk_unlock_cpuirqrs(&bma->lock, flag);
345 rc = XWOK;
346 *membuf = xwmm_bma_bcb_to_mem(bma, bcb);
347 }
348 return rc;
349}
static struct xwmm_bma_bcb * xwmm_bma_orderlist_choose(struct xwmm_bma *bma, struct xwmm_bma_orderlist *ol)
从阶链表中选择一块内存,并返回其块控制块
Definition bma.c:244
#define xwmm_bmalogf(lv, fmt,...)
Definition bma.c:25
static void * xwmm_bma_bcb_to_mem(struct xwmm_bma *bma, struct xwmm_bma_bcb *bcb)
从内存块的控制块指针获得内存块的首地址
Definition bma.c:138
static void xwmm_bma_divide_block(struct xwmm_bma *bma, struct xwmm_bma_bcb *bcb, xwsq_t target_odr, struct xwmm_bma_orderlist *curr_ol)
将大内存块分割成小块
Definition bma.c:276
#define ENOENT
No such file or directory
Definition errno.h:32
#define EFAULT
Bad address
Definition errno.h:44
#define ENOMEM
Not enough space
Definition errno.h:42
#define XWOK
No error
Definition errno.h:182
#define ERANGE
Result too large
Definition errno.h:64
static __xwcc_inline void *__xwcc_must_check err_ptr(xwer_t err)
将错误码转换为指针
Definition error.h:42
static __xwcc_inline bool __xwcc_must_check is_err(const void *ptr)
测试指针的值是否为错误码
Definition error.h:65
signed long xwer_t
Definition type.h:554
#define NULL
Definition type.h:28
unsigned long xwsq_t
Definition type.h:445
#define XWSSQ_MAX
Definition type.h:473
unsigned long xwptr_t
Definition type.h:375
xwptr_t xwreg_t
Definition type.h:409
static void xwos_splk_unlock_cpuirqrs(struct xwos_splk *spl, xwreg_t cpuirq)
XWOS API:解锁自旋锁,并恢复本地CPU的中断标志
Definition spinlock.h:224
static void xwos_splk_lock_cpuirqsv(struct xwos_splk *spl, xwreg_t *cpuirq)
XWOS API:上锁自旋锁,保存本地CPU的中断标志并关闭
Definition spinlock.h:192
#define XWOS_VALIDATE(exp, errstr,...)
检查函数参数是否有效
Definition standard.h:76
xwu8_t order
Definition bma.h:71
struct xwos_splk lock
Definition bma.h:85
struct xwmm_bma_orderlist * orderlists
Definition bma.h:86
struct xwmm_bma_bcb * bcbs
Definition bma.h:87
函数调用图:

◆ xwmm_bma_free()

xwer_t xwmm_bma_free ( struct xwmm_bma bma,
void *  mem 
)

XWMM API:释放内存块

参数
[in]bma伙伴算法内存块分配器对象的指针
[in]mem内存块的首地址指针
返回
错误码
返回值
XWOK没有错误
-EINVAL参数错误
-ERANGE内存块不属于指定的伙伴算法内存块分配器对象
注解
  • 同步/异步:同步
  • 上下文:中断、中断底半部、线程
  • 重入性:可重入

< 块正在被使用

<No error

在文件 bma.c402 行定义.

403{
404 struct xwmm_bma_bcb * bcb;
405 xwreg_t flag;
406 xwer_t rc;
407
408 XWOS_VALIDATE((bma), "nullptr", -EFAULT);
409 XWOS_VALIDATE((mem), "nullptr", -EFAULT);
410
411 if ((((xwptr_t)mem < bma->zone.origin) ||
412 ((xwptr_t)mem >= (bma->zone.origin + bma->zone.size)))) {
413 rc = -ERANGE;
414 goto err_range;
415 }
416 bcb = xwmm_bma_mem_to_bcb(bma, mem);
417 if (is_err(bcb)) { // cppcheck-suppress [misra-c2012-14.4]
418 rc = ptr_err(bcb);
419 goto err_invalmem;
420 }
421 xwmm_bmalogf(DEBUG,
422 "[FREE] mem:0x%lX,bcb(idx:0x%lX,odr:0x%X)\n",
423 (xwptr_t)mem,
424 (((xwptr_t)bcb - (xwptr_t)xwmm_bma->bcbs) /
425 sizeof(struct xwmm_bma_bcb)),
426 bcb->order);
427 if (0 == (XWMM_BMA_INUSED & bcb->order)) {
428 rc = -EINVAL;
429 goto err_invalmem;
430 }
431 xwos_splk_lock_cpuirqsv(&bma->lock, &flag);
432 xwmm_bma_combine(bma, bcb);
433 xwos_splk_unlock_cpuirqrs(&bma->lock, flag);
434 return XWOK;
435
436err_invalmem:
437err_range:
438 return rc;
439}
static struct xwmm_bma_bcb * xwmm_bma_mem_to_bcb(struct xwmm_bma *bma, void *mem)
从内存块首地址获得其控制块的指针
Definition bma.c:115
static void xwmm_bma_combine(struct xwmm_bma *bma, struct xwmm_bma_bcb *bcb)
合并内存块
Definition bma.c:358
#define XWMM_BMA_INUSED
Definition bma.h:47
#define EINVAL
Invalid argument
Definition errno.h:52
static __xwcc_inline xwer_t __xwcc_must_check ptr_err(const void *ptr)
将指针的值转换为错误码
Definition error.h:53
struct xwmm_zone zone
Definition bma.h:81
xwptr_t origin
Definition common.h:41
xwsz_t size
Definition common.h:42
函数调用图:

◆ xwmm_bma_init()

xwer_t xwmm_bma_init ( struct xwmm_bma bma,
const char *  name,
xwptr_t  origin,
xwsz_t  size,
xwsz_t  blksize,
xwsz_t  blkodr 
)

XWMM API:初始化伙伴算法内存块分配器

参数
[in]bma伙伴算法内存块分配器的指针
[in]name名字
[in]origin内存区域的起始地址
[in]size内存区域的大小
[in]blksize伙伴算法内存块分配器中单位内存块的大小
[in]blkodr伙伴算法内存块分配器中单位内存块的数量,以2的blkodr次方形式表示
返回
错误码
返回值
-ESIZE内存区域大小不匹配
注解
  • 单位内存块的数量只能是2的n次方,即 2, 4, 8, 16, 32, 64, 128, ... ,对应的 blkodr 分别为 1, 2, 3, 4, 5, 6, 7, ...
  • 内存区域大小必须满足关系: size == (blksize * (1 << blkodr))
  • 同步/异步:同步
  • 上下文:中断、中断底半部、线程
  • 重入性:不可重入

<Size error

< 块正在被使用

< 块已被合并

< 块正在被使用

<No error

在文件 bma.c62 行定义.

65{
66 xwer_t rc;
67 xwsz_t num;
68 xwsz_t i;
69
70 XWOS_VALIDATE((bma), "nullptr", -EFAULT);
71
72 num = 1U << blkodr;
73 if (size != (num * blksize)) {
74 rc = -ESIZE;
75 xwmm_bmalogf(ERR, "Size of memory(0x%lX, 0x%lX) is error!\n",
76 origin, size);
77 goto err_size;
78 }
79 bma->name = name;
80 bma->zone.origin = origin;
81 bma->zone.size = blksize * num;
82 bma->blksize = blksize;
83 bma->blkodr = blkodr;
84 xwos_splk_init(&bma->lock);
85 bma->orderlists = (struct xwmm_bma_orderlist *)&bma[(xwsz_t)1];
86 bma->bcbs = (struct xwmm_bma_bcb *)&bma->orderlists[(xwsz_t)1 + blkodr];
87 xwmm_bmalogf(DEBUG,
88 "memory:(0x%lX,0x%lX),orderlists:0x%lX,bcbs:0x%lX,"
89 "blocksize:0x%lX, blockorder:0x%lX\n",
90 origin, size, (xwptr_t)bma->orderlists, (xwptr_t)bma->bcbs,
91 bma->blksize, bma->blkodr);
92
93 for (i = 0; i < num; i++) {
95 }
96 bma->bcbs[0].order = (xwu8_t)blkodr | XWMM_BMA_INUSED;
97
98 for (i = 0; i <= blkodr; i++) {
100 }
101 xwmm_bma_orderlist_add(bma, &bma->orderlists[blkodr], blkodr, &bma->bcbs[0]);
102 return XWOK;
103
104err_size:
105 return rc;
106}
static void xwmm_bma_orderlist_add(struct xwmm_bma *bma, struct xwmm_bma_orderlist *ol, xwu8_t odr, struct xwmm_bma_bcb *bcb)
将一块内存加入到阶链表
Definition bma.c:175
#define XWMM_BMA_COMBINED
Definition bma.h:45
static void xwlib_bclst_init_head(struct xwlib_bclst_node *h)
初始化一个链表头。
Definition bclst.h:229
#define ESIZE
Size error
Definition errno.h:199
unsigned long xwsz_t
Definition type.h:339
static void xwos_splk_init(struct xwos_splk *spl)
XWOS API:初始化自旋锁
Definition spinlock.h:89
struct xwlib_bclst_node head
Definition bma.h:64
const char * name
Definition bma.h:82
xwsz_t blksize
Definition bma.h:83
函数调用图: