程序清单 L7.3 OSMemCreate()
| OS_MEM *OSMemCreate (void *addr, INT32U nblks, INT32U blksize, INT8U *err)
| {
| OS_MEM *pmem;
| INT8U *pblk;
| void **plink;
| INT32U i;
|
|
| if (nblks < 2) { (1)
| *err = OS_MEM_INVALID_BLKS;
| return ((OS_MEM *)0);
| }
| if (blksize < sizeof(void *)) { (2)
| *err = OS_MEM_INVALID_SIZE;
| return ((OS_MEM *)0);
| }
| OS_ENTER_CRITICAL();
| pmem = OSMemFreeList; (3)
| if (OSMemFreeList != (OS_MEM *)0) {
| OSMemFreeList = (OS_MEM *)OSMemFreeList->OSMemFreeList;
| }
| OS_EXIT_CRITICAL();
| if (pmem == (OS_MEM *)0) { (4)
| *err = OS_MEM_INVALID_PART;
| return ((OS_MEM *)0);
| }
| plink = (void **)addr; (5)
| pblk = (INT8U *)addr + blksize;
| for (i = 0; i < (nblks - 1); i++) {
| *plink = (void *)pblk;
| plink = (void **)pblk;
| pblk = pblk + blksize;
| }
| *plink = (void *)0;
| OS_ENTER_CRITICAL();
| pmem->OSMemAddr = addr; (6)
| pmem->OSMemFreeList = addr;
| pmem->OSMemNFree = nblks;
| pmem->OSMemNBlks = nblks;
| pmem->OSMemBlkSize = blksize;
| OS_EXIT_CRITICAL();
| *err = OS_NO_ERR;
| return (pmem); (7)
| }
|
图 F7.4是OSMemCreate()函数完成后,内存控制块及对应的内存分区和分区内的内存块之间的关系。在程序运行期间,经过多次的内存分配和释放后,同一分区内的各内存块之间的链接顺序会发生很大的变化。
分配一个内存块,OSMemGet()应用程序可以调用OSMemGet()函数从已经建立的内存分区中申请一个内存块。该函数的唯一参数是指向特定内存分区的指针,该指针在建立内存分区时,由OSMemCreate()函数返回。显然,应用程序必须知道内存块的大小,并且在使用时不能超过该容量。例如,如果一个内存分区内的内存块为32字节,那么,应用程序最多只能使用该内存块中的32字节。当应用程序不再使用这个内存块后,必须及时把它释放,重新放入相应的内存分区中[见7.03节,释放一个内存块,OSMemPut()]。
图 F7.4 OSMemCreate()——Figure 7.4
程序清单 L7.4是OSMemGet()函数的源代码。参数中的指针pmem指向用户希望从其中分配内存块的内存分区[L7.4(1)]。OSMemGet()首先检查内存分区中是否有空闲的内存块[L7.4(2)]。如果有,从空闲内存块链表中删除第一个内存块[L7.4(3)],并对空闲内存块链表作相应的修改[L7.4(4)]。这包括将链表头指针后移一个元素和空闲内存块数减1[L7.4(5)]。最后,返回指向被分配内存块的指针[L7.4(6)]。
程序清单 L7.4 OSMemGet()
| void *OSMemGet (OS_MEM *pmem, INT8U *err) (1)
| {
| void *pblk;
|
|
| OS_ENTER_CRITICAL();
| if (pmem->OSMemNFree > 0) { (2)
| pblk = pmem->OSMemFreeList; (3)
| pmem->OSMemFreeList = *(void **)pblk; (4)
| pmem->OSMemNFree--; (5)
| OS_EXIT_CRITICAL();
| *err = OS_NO_ERR;
| return (pblk); (6)
| } else {
| OS_EXIT_CRITICAL();
| *err = OS_MEM_NO_FREE_BLKS;
| return ((void *)0);
| }
| }
|
值得注意的是,用户可以在中断服务子程序中调用OSMemGet(),因为在暂时没有内存块可用的情况下,OSMemGet()不会等待,而是马上返回NULL指针。 |