相关文章
.Net Micro Framework 模拟器详解 -- 内存配置
摘要:对嵌入式设备来说,存储空间的大小无疑是一个很重要的参数,但是调试的时候我们通常无法方便地对不同内存大小的设备进行测试。这时候使用模拟器是一个很好的选择,本文介绍了如何在.Net Micro Framework中通过代码和XML配置文件两种方式来配置模拟器的Memory(包括Ram和Flash Memory)。并使得模拟器具有持久存储的能力。
Keywords:
.Net Micro Framework, Hardware Emulator, RAM, FLASH, XML,Configuration
在.Net Micro Framework中,提供了MemoryManager 类型(见Microsoft.SPOT.Emulator.Memory命名空间)来描述和模拟RAM和FLASH。不过这个类型本身并不直接提供属性和方法来对模拟器的存储空间进行配置,而是提供了以下两个属性:RamManager和FlashManager 。这两个属性同属于微软提供的模拟器组件(Emulator Components)。它们都是由Memorymanager跟据你的配置动态创建的。你可以在Emulator的配置文件里面来完成RAM和FLASH的配置,这我们将在后文讨论。
Random Access Memory (RAM)
前面提到的MemoryManager.RamManager是一个同名类型(RamManager)的属性,对于RamManager类型,唯一你可以设置(set)的属性就是Size,它表示了RAM的大小,单位是byte。在配置文件里面,你需要先在”<Types>”的标签区域内将MemoryManager 和RamManager标签添加进来,这就像C#代码中的using一样,使得EmulatorComponents内的MemoryManager 和RamManager标签能被模拟器的配置引擎解析出来。下面的XML代码演示了如何在配置文件里以一个16进制数0x2000000来设置32兆的RAM:
需要注意的是,模拟器的RAM默认情况下只有1Mbyte
之前我尝试在带有VGA(640x480)的LCD模拟器上尝试显示大图的时候,Visual Studio会抛出以下异常:
Failed allocation for 51205 blocks, 614460 bytes
在System.OutOfMemoryException 中第一次偶然出现的“Microsoft.SPOT.Graphics.dll”类型的异常
未处理的“System.OutOfMemoryException”类型的异常出现在Microsoft.SPOT.Graphics.dll 中。
在我把模拟器的内存配置为32兆之后就不再出现这个问题了。这是因为模拟器默认的RAM大小为1024kb.而我的图片640*480*16/8=bmp600kb+图片77kb+应用用程序大小已经超过了RAM的分配限制。
FLASH
和RamManager类似,FlashManager同样以MemoryManager的一个属性的方式供开发人员来配置。相对RAM来说FLASH的存储区域显得不是那么单纯。FLASH的存储单位(包括读和写)是扇区 (sector)。由一个或多个扇区构成了功能不同的段(block),用来存放不同的内容。而通常我们说的一个分区(partition)又是由一个或多个段构成的,不过Micro Framework设备的Flash目前均只有一个分区(暂时只支持一个分区), 所以即将推出的MF文件系统(.Net MF 3.0 feature)和Windows Mobile一样,都将只有一个根目录索引,而不会和Windows一样还有盘符之类的标识。FlashManager类用来管理sector,在Emulator.config中,每一对FlashSector标签都对应着这样的一个结构:
和RamManager不同的是,对FlashManager来说,Size属性是不需要制定的,因为它的存储空间大小会由你配置的Sectors自动计算出来,除了Size之外它还具有以下可以设置的属性:
• bool IsReadOnly
• uint MaxSectorEraseTime
• uint MaxWordWriteTime
FLASH的第一个sector必须将Partition和Block设为Start(或者StartEnd, 如果该sector为Partition或Block包含的唯一sector)。最后一个sector必须将Partition和Block配置为End。总之,Start和End总是成对存在的,表示一个sector是某个分区或者段的开始和结束。 否则在构造模拟器的时候Visual Studio会抛出异常,例如:
值得一提的是FlashSector 的Usage 属性,它是FlashSectorUsage枚举类型,它的可用成员如下表:
| 成员名称 | 描述 |
| 指明当前的sector用来存放application code. | |
| 指明当前的sector用来存放 bootstrap code. | |
| 指明当前的sector用来存放 common language runtime (CLR) code. | |
| 指明当前的sector用来存放配置信息. | |
| 指明当前的sector用来存放 client application code. | |
| 指明当前的sector用来存放操作记录. | |
| 指明当前的sector用来存放 programmer-defined data. | |
| 指明当前的sector用来存放 programmer-defined data. |
这些描述信息都将被.Net Micro Framework用于优化和校验程序。对应的XML配置文件(Emulator.config)如下:
首先在”<Types>”标签内加入以下两行声明:
<FlashManager>Microsoft.SPOT.Emulator.Memory.FlashManager</FlashManager>
<FlashSector>Microsoft.SPOT.Emulator.Memory.FlashSector</FlashSector>
然后就可以在<MemoryManager>中进行配置了:
注意这里实际上模拟器的FlashSector总的大小是1024kb,可以参考:
真正的模拟Flash – 持久存储
实际上在前面提到的Flash并不是严格意义上的Flash,毕竟只是模拟。而模拟器本身又是一个在PC机的内存中运行的一个程序,在你关闭了模拟器之后,你存在模拟器的Flash中的数据还是会丢失。不过我们在.Net Micro Framework的SDK中可以找到一个叫”EWRSampleEmulator”的Sample,它在”Microsoft .NET Micro Framework"Samples"ExtendedWeakReferences”目录下。这个Sample模拟器在关闭时会将Flash中的内容写到一个本地文件里面去。在模拟器下一次启动的时候,会从这些存在PC机上的数据文件去初始化模拟器,从而模拟Flash持久存储数据的特征。下面我们通过一个继承自FlashManager的类看看它是如何实现这个持久存储的过程的:
使用这个类配置模拟器的时候,需要在“<Types>”标签里面先声明这个类以及它所在的程序集名称。简单来说就是把配置文件默认的FlashManager声明:
<FlashManager>Microsoft.SPOT.Emulator.Memory.FlashManager</FlashManager>
用类似:
<FlashManager>MyNameSpace.PersistentFlashManager,
MyAssemblyName</FlashManager>
替换掉即可。
Enjoy!
黄季冬