OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 9681|回复: 41

内存管理的疑问?内存管理表初始化的时候,为什么要乘以2???

[复制链接]

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
发表于 2014-5-11 22:38:55 | 显示全部楼层 |阅读模式
5金钱
//内存管理表
u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]; //内部SRAM内存池MAP
u16 mem2mapbase[MEM2_ALLOC_TABLE_SIZE] __attribute__((at(0X68000000+MEM2_MAX_SIZE))); //外部SRAM内存池MAP
//内存管理参数   
const u32 memtblsize[2]={MEM1_ALLOC_TABLE_SIZE,MEM2_ALLOC_TABLE_SIZE}; //内存表大小 0---1279,0---6143
const u32 memblksize[2]={MEM1_BLOCK_SIZE,MEM2_BLOCK_SIZE}; //内存分块大小
const u32 memsize[2]={MEM1_MAX_SIZE,MEM2_MAX_SIZE}; //内存总大小

//设置内存
//*s:内存首地址
//c :要设置的值
//count:需要设置的内存大小(字节为单位)
void mymemset(void *s,u8 c,u32 count)  
{  
    u8 *xs = s;     //这里是隐式转换,把void*转换为u8*
    while(count--)*xs++=c;  
}   
//内存管理初始化  
//memx:所属内存块
void mem_init(u8 memx)  
{  
    mymemset(mallco_dev.memmap[memx], 0,memtblsize[memx]*2);//内存状态表数据清零  
    mymemset(mallco_dev.membase[memx], 0,memsize[memx]); //内存池所有数据清零  
    mallco_dev.memrdy[memx]=1; //内存管理初始化OK  
}
那个memtblsize数组不是32位的吗???还要乘以2,那内存管理表的编号岂不是扩大2倍了???

最佳答案

查看完整内容[请看2#楼]

不知道我的这样理解是否是对的?? //内存管理控制器 struct _m_mallco_dev { void (*init)(u8); //初始化 u8 (*perused)(u8);       //内存使用率 u8 *membase[2]; //内存池 管理2个区域的内存 u16 *memmap[2]; //内存管理状态表 u8  memrdy[2]; //内存管理是否就绪 }; 因为u16 *memmap[2];是u16的指针数组,而在mymemset(void *s,u8 c,u32 count)函数中把传入的内存管理表 ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-11 22:38:56 | 显示全部楼层
不知道我的这样理解是否是对的??
//内存管理控制器
struct _m_mallco_dev
{
void (*init)(u8); //初始化
u8 (*perused)(u8);       //内存使用率
u8 *membase[2]; //内存池 管理2个区域的内存
u16 *memmap[2]; //内存管理状态表
u8  memrdy[2]; //内存管理是否就绪
};
因为u16 *memmap[2];是u16的指针数组,而在mymemset(void *s,u8 c,u32 count)函数中把传入的内存管理表的地址强制转换为u8*的类型,那么强制转换为u8*之后的地址是指向传入地址的低8位,为了能够也把传入地址的高位也设置为0,那么内存管理表的值必须是要乘以2的,这样才可以刚刚好由u8*类型的地址初始化完u16*地址的数据。。。应该是这样子的。。。。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-11 22:55:27 | 显示全部楼层
mymemset(mallco_dev.memmap[memx], 0,memtblsize[memx]*2);//内存状态表数据清零  
这句话的意思(作用),麻烦您用中文翻译一下,念出来。
然后我再告诉你问题在哪里。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-11 23:02:45 | 显示全部楼层
回复【2楼】正点原子:
---------------------------------
把u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE];这个数组的内容清零。。。。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-11 23:52:07 | 显示全部楼层
回复【4楼】STM32初学菜鸟:
---------------------------------
2楼你已经回答了自己的问题了。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-12 15:17:01 | 显示全部楼层
回复【5楼】正点原子:
---------------------------------
谢谢原子哥。。。。
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-12 16:16:27 | 显示全部楼层
回复【5楼】正点原子:
---------------------------------
原子哥,如果把这个函数的u8*类型改为和数组一样都是u16的话,应该不需要乘以2了,看到原子哥你之前回复别人的,不解,我写代码测试了一下,好像也是不需要的。。。。
void mymemset(void *s,u8 c,u32 count)    
{    
    u16*xs = s;             //u8*xs = s;
    while(count--)*xs++=c;    
}
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-13 00:13:50 | 显示全部楼层
回复【7楼】STM32初学菜鸟:
---------------------------------
是可以啊,但是你这样就不标准了...
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-13 18:14:25 | 显示全部楼层
回复【8楼】正点原子:
---------------------------------
原子哥,不标准的意思是指啥?
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-13 20:28:19 | 显示全部楼层
回复【9楼】STM32初学菜鸟:
---------------------------------
百度,c语言标准的memset函数,操作是以1个字节为单位,还是以2字节为单位?
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

23

主题

81

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
343
金钱
343
注册时间
2013-10-28
在线时间
0 小时
 楼主| 发表于 2014-5-13 22:56:11 | 显示全部楼层
回复【10楼】正点原子:
---------------------------------
查了一下还真是以字节来错操作的,STM32如果内存泄露了,复位或者重启是否会回收呢???
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-13 23:17:21 | 显示全部楼层
回复【11楼】STM32初学菜鸟:
---------------------------------
复位/重启,就不叫回收了...
这叫从新开始.
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 16:35:17 | 显示全部楼层
回复【10楼】正点原子:
---------------------------------
请问内存分配方向是从低位到高位吗?
回复

使用道具 举报

70

主题

6763

帖子

0

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
13129
金钱
13129
注册时间
2012-11-26
在线时间
3814 小时
发表于 2014-5-18 18:17:58 | 显示全部楼层
回复【13楼】chengmingluo:
---------------------------------
没记错的话  是从高到低
学无止境
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-5-18 21:19:21 | 显示全部楼层
回复【13楼】chengmingluo:
---------------------------------
从高到低.
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 21:20:19 | 显示全部楼层
回复【14楼】jermy_z:
---------------------------------
但是mymemset(void *s,u8 c,u32 count)这个函数不是从低8位开始吗
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 21:50:00 | 显示全部楼层
回复【15楼】正点原子:
---------------------------------
但是mymemset(void *s,u8 c,u32 count)这个函数不是从低8位开始吗
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 21:58:59 | 显示全部楼层
回复【15楼】正点原子:
---------------------------------
还有,为什么为什么内存管理表是u16
回复

使用道具 举报

70

主题

6763

帖子

0

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
13129
金钱
13129
注册时间
2012-11-26
在线时间
3814 小时
发表于 2014-5-18 22:16:15 | 显示全部楼层
回复【16楼】chengmingluo:
---------------------------------
那只是把C赋值给S而已,初始化值而已,与分配内存又没关系
学无止境
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 22:42:13 | 显示全部楼层
回复【19楼】jermy_z:
---------------------------------
那假设s是00000001 11111111,转化后为11111111,地址已经不同了,那不是赋错值了?
回复

使用道具 举报

70

主题

6763

帖子

0

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
13129
金钱
13129
注册时间
2012-11-26
在线时间
3814 小时
发表于 2014-5-18 22:44:34 | 显示全部楼层
回复【20楼】chengmingluo:
---------------------------------

你还是去看看memset是干嘛的吧!!
学无止境
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-18 22:57:13 | 显示全部楼层
回复【21楼】jermy_z:
---------------------------------
把u8类型的数据“c”填充到u16类型指针元素memmap[]中
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-19 21:17:09 | 显示全部楼层
回复【21楼】jermy_z:
---------------------------------
回复【21楼】jermy_z:
---------------------------------
网上有两种格式,一种是memset(f,2,sizeof(int)*n),一种是memset(void *s,int ch, unsigned n),都一样的吗
回复

使用道具 举报

12

主题

87

帖子

0

精华

初级会员

Rank: 2

积分
155
金钱
155
注册时间
2014-4-2
在线时间
0 小时
发表于 2014-5-20 23:12:14 | 显示全部楼层
回复【23楼】chengmingluo:
---------------------------------
memset

将s所指向的某一块内存中的前n个 字节的内容全部设置为ch指定的ASCII值, 块的大小由第三个参数指定,这个函数通常为新申请的内存做初始化工作, 其返回值为指向S的指针。
回复

使用道具 举报

68

主题

136

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
428
金钱
428
注册时间
2014-6-27
在线时间
8 小时
发表于 2014-12-27 14:01:04 | 显示全部楼层
大赞,力顶楼主,这个问题我也想了好久!
回复

使用道具 举报

15

主题

72

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
348
金钱
348
注册时间
2015-5-2
在线时间
29 小时
发表于 2015-5-2 22:18:16 | 显示全部楼层
请教原子哥,
// 释放内存(外部调用) 
// memx:所属内存块
// ptr:内存首地址 
void myfree(u8 memx, void *ptr)
{
u32 offset;
// 地址为0
if(ptr == NULL) return;
offset = (u32)ptr - (u32)malloc_dev.membase[memx];
mem_free(memx, offset);
}


// 重新分配内存(外部调用)
// memx:所属内存块
// *ptr:旧内存首地址
// size:要分配的内存大小(字节)
// 返回值:新分配到的内存首地址.
void *myrealloc(u8 memx, void *ptr, u32 size)
{
u32 offset;
offset = mem_malloc(memx, size);
if(offset == 0xFFFFFFFF)
return NULL;
else
{
// 拷贝旧内存内容到新内存
mymemcpy((void*)((u32)malloc_dev.membase[memx] + offset), ptr, size);
// 释放旧内存
myfree(memx, ptr);
// 返回新内存首地址
return ((void*)((u32)malloc_dev.membase[memx] + offset));
}
}


myfree函数的偏移地址是offset = (u32)ptr - (u32)malloc_dev.membase[memx];
myrealloc的偏移地址是(u32)malloc_dev.membase[memx] + offset)
想了好久不知道为啥是这样啊?!

回复

使用道具 举报

4

主题

8

帖子

0

精华

新手上路

积分
44
金钱
44
注册时间
2015-3-25
在线时间
0 小时
发表于 2015-5-17 22:38:01 | 显示全部楼层
回复【26楼】ndzhzsdw:
---------------------------------
(u32)malloc_dev.membase[memx] + offset)这个不是偏移地址,是分配新内存后返回的地址,offset才表示偏移地址。
回复

使用道具 举报

15

主题

72

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
348
金钱
348
注册时间
2015-5-2
在线时间
29 小时
发表于 2015-5-18 10:06:34 | 显示全部楼层
回复【27楼】kecai:
---------------------------------
嗯嗯  谢谢
回复

使用道具 举报

2

主题

18

帖子

0

精华

新手上路

积分
47
金钱
47
注册时间
2015-2-2
在线时间
0 小时
发表于 2015-5-25 17:33:15 | 显示全部楼层
回复【2楼】正点原子:
---------------------------------
原子兄,探索者内存管理实验中有如下代码对STM32F4内部112K内存块进行管理:
#define MEM1_BLOCK_SIZE  32            //内存块大小为 32 字节
#define MEM1_MAX_SIZE 100*1024     //最大管理内存 100K
#define MEM1_ALLOC_TABLE_SIZE  MEM1_MAX_SIZE/MEM1_BLOCK_SIZE
由上,可以计算出内存管理表的大小是100 * 1024 / 32 = 3200(个),一个内存管理表项用两个字节来存储,则内存管理表占用的总大小是3200 * 2 = 6400(字节) = 6.25K;
我的问题来了,112K - 100K - 6.25K = 5.75K,这剩下的5.75K内存为何没有作管理,还是我的理解哪里有错?请指教~~
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2015-5-25 17:44:05 | 显示全部楼层
回复【29楼】YouDream:
---------------------------------
你的全局变量,堆栈,这些都要用内存啊
另外,你剩一点内存还是好的,指不定下次加几行代码,咋办?
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

2

主题

18

帖子

0

精华

新手上路

积分
47
金钱
47
注册时间
2015-2-2
在线时间
0 小时
发表于 2015-5-25 18:02:32 | 显示全部楼层
回复【30楼】正点原子:
---------------------------------
还有一个疑问,
//内存池(32字节对齐)
__align(32) u8 mem1base[MEM1_MAX_SIZE]; //内部SRAM内存池
__align(32) u8 mem2base[MEM2_MAX_SIZE] __attribute__((at(0X68000000))); //外部SRAM内存池
__align(32) u8 mem3base[MEM3_MAX_SIZE] __attribute__((at(0X10000000))); //内部CCM内存池
//内存管理表
u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]; //内部SRAM内存池MAP
u16 mem2mapbase[MEM2_ALLOC_TABLE_SIZE] __attribute__((at(0X68000000+MEM2_MAX_SIZE))); //外部SRAM内存池MAP
u16 mem3mapbase[MEM3_ALLOC_TABLE_SIZE] __attribute__((at(0X10000000+MEM3_MAX_SIZE))); //内部CCM内存池MAP
为啥其它两个内存池指定了绝对地址访问,而内部112K内存池就没有指定呢?谢谢~~
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2015-5-26 00:09:51 | 显示全部楼层
回复【31楼】YouDream:
---------------------------------
因为内部128K,是有MDK自动分配,故不需要设置地址。
而外部sram和CCM,都是需要指定地址的,否则就按内部sram处理。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

2

主题

18

帖子

0

精华

新手上路

积分
47
金钱
47
注册时间
2015-2-2
在线时间
0 小时
发表于 2015-5-26 08:48:21 | 显示全部楼层
回复【32楼】正点原子:
---------------------------------
谢谢原子兄的解答~~
那如果我指定了地址,
__align(32) u8 mem1base[MEM1_MAX_SIZE]        __attribute__((at(0X20000000))); //内部SRAM内存池
u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE]   __attribute__((at(0X20000000+MEM1_MAX_SIZE))); //内部SRAM内存池MAP
也是没有问题的吧?
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2015-5-27 01:07:39 | 显示全部楼层
回复【33楼】YouDream:
---------------------------------
可以,只要编译器不报错。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

36

主题

195

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
313
金钱
313
注册时间
2014-10-27
在线时间
34 小时
发表于 2015-6-4 16:15:36 | 显示全部楼层
学习了,mark
回复

使用道具 举报

22

主题

104

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
235
金钱
235
注册时间
2015-5-4
在线时间
34 小时
发表于 2015-11-2 16:53:25 | 显示全部楼层
学习了,mark
回复

使用道具 举报

15

主题

143

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
243
金钱
243
注册时间
2014-10-31
在线时间
4 小时
发表于 2016-1-3 20:46:19 | 显示全部楼层
又学了些知识:,mark
总有一天,要为自由而活
回复

使用道具 举报

9

主题

35

帖子

0

精华

初级会员

Rank: 2

积分
105
金钱
105
注册时间
2013-10-24
在线时间
10 小时
发表于 2016-1-26 11:37:33 | 显示全部楼层
其实很多人不明白,觉得*2好像对应的地址数量变成了两倍,一定要清楚,最后的count只是提示要执行多少次循环。
回复

使用道具 举报

0

主题

1

帖子

0

精华

新手入门

积分
26
金钱
26
注册时间
2015-11-16
在线时间
1 小时
发表于 2016-6-2 21:21:15 | 显示全部楼层
将内存池分为n个内存块对应内存管理表也分为n份  就是一个内存块对应管理表的一份
而内存管理表的一份需两个字节u16 mem1mapbase[MEM1_ALLOC_TABLE_SIZE];
例如你要初始化10个内存块,你要将对应的内存表中连续的十个字节设置为0
回复

使用道具 举报

1

主题

3

帖子

0

精华

新手入门

积分
8
金钱
8
注册时间
2016-6-3
在线时间
2 小时
发表于 2016-6-3 09:10:03 | 显示全部楼层
学习了
回复

使用道具 举报

4

主题

11

帖子

0

精华

新手上路

积分
49
金钱
49
注册时间
2016-5-21
在线时间
14 小时
发表于 2016-9-25 16:27:14 | 显示全部楼层
初级选手 发表于 2016-1-26 11:37
其实很多人不明白,觉得*2好像对应的地址数量变成了两倍,一定要清楚,最后的count只是提示要执行多 ...

这位兄台,回答简单明了!一语中的。。。。
回复

使用道具 举报

19

主题

169

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
409
金钱
409
注册时间
2018-4-21
在线时间
107 小时
发表于 2019-1-9 21:45:07 | 显示全部楼层
学习了,看内存代码没头绪,这样一看可以理解
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-23 07:18

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表