其实要把特别大的变量(数组)定义到SDRAM中步骤很简单,但这个过程却困扰了我好久,此篇文章就作为个人学习笔记参考参考吧。
一、开启SDRAM
在Env中输入menuconfig进入菜单,找到Enable SDRAM,开启该bsp
(资料图片仅供参考)
可以看到在Drivers下多了一个drv_sdram.c文件
编译成功后可以看一下SDRAM的各项参数,在sdram_port.h文件中
查阅资料时,都提到如果要使用SDRAM,必须要在进入 __main 前对SDRAM进行初始化,但是RTT很人性化的直接解决了这个问题,这个BSP已经实现了初始化这一步。可以看看串口打印的数据:
sdram init success, mapped at 0xC0000000, size is 33554432 bytes, data width is 16 初始化已经成功,首地址、大小以及数据长度的信息都已经打印出来,所以不需要画蛇添足地再在start_up.s文件中添加SDRAM_Init函数
二、使用memheap申请和管理SDRAM空间
这一步要在Env中开启Use all of memheap objects as heap
开启后生成新工程,实际上就是开启了RT_USING_MEMHEAP_AS_HEAP
三、定义和使用变量
我们可以使用rt_memheap_alloc来直接申请SDRAM内的空间,直接上代码
struct rt_memheap system_heap; #define LED0 GET_PIN(B,1) void sdram_entry() { rt_uint8_t key; rt_uint8_t i=0,led0sta=1; rt_uint32_t ts=0,ty=0; rt_uint16_t * testsram = RT_NULL; testsram = rt_memheap_alloc(&system_heap, 2050*1024); led_Init(); key_init(); //SDRAM_INIT();//RTT已经初始化了 for(ts=0;ts<800;ts++) { for(ty=0;ty<1280;ty++) { testsram[ts*ty]=ty*ts; } } while(1) { key=key_scan(0);//不支持连按 if(key==KEY1_PRES)//打印预存测试数据 { for(ts=0;ts<1280*800;ts++) { rt_kprintf("testsram[%d]:%drn",ts,testsram[ts]);//显示测试数据 ts+=200; } }else rt_thread_mdelay(10); i++; if(i==20)//DS0闪烁. { i=0; led0sta=!led0sta; rt_pin_write(LED0,led0sta); } } } #define THREAD_STACK_SIZE 440 #define THRAED_PRIORITY 20 #define THREAD_TIMESKICE 10 int sdram_sample(void) { rt_thread_t tid = RT_NULL; tid = rt_thread_create("sdram_thread",sdram_entry(void*)1,THREAD_STACK_SIZE,THRAED_PRIORITY,THREAD_TIMESKICE); if(tid!=RT_NULL)rt_thread_startup(tid); return 0; } MSH_CMD_EXPORT(sdram_sample,sdram sample); struct rt_memheap system_heap;
考虑到SDRAM初始化的问题,这一句主要是把初始化好的SDRAM空间用上,我在drv_sdram.c中将system_heap设为了外部变量
这样就可以在其他文件中继续使用这个初始化好的SDRAM。
接下来我主要是想定义一个16位1280*800大小的数组,是1280 乘以 800 乘以 2字节(16位为2字节)再除以1024等于2000KBytes,多算一点2050KBytes。
代码中的按键和LED相关代码可以删除,要验证的话可以直接看这三处:
rt_uint16_t * testsram = RT_NULL; testsram = rt_memheap_alloc(&system_heap, 2050*1024); for(ts=0;ts<800;ts++) { for(ty=0;ty<1280;ty++) { testsram[ts*ty]=ty*ts; } } for(ts=0;ts<1280*800;ts++) { rt_kprintf("testsram[%d]:%drn",ts,testsram[ts]);//显示测试数据 ts+=200; }
四、烧写到板子上,验证结果
在调用sdram_sample前使用命令list_memheap可以看到:
SDRAM初始化成功,大小和使用量都有,我们可以看到片外的 SDRAM 初始化之后我们并没有使用,但是在 max used size 字段中确显示已经使用了 48 字节的空间,这部分空间是内存堆的数据头,用于 magic、used 信息及链表节点使用。
使用sdram_sample后,可以看到打印数据:
由于定义的是16位,最大为65535,所以后面溢出了。
最后打印完毕(因为一个一个打印太慢了,所以我加了“ts+=200”的语句,跳着打印),再使用list_memheap可以看到:
SDRAM内的空间已经使用了一部分,而内部RAM使用量没有变。2050KBytes乘以1024=2099200字节,这就是我们申请使用的空间。
关键词: