.Net Micro Framework研究—FAT文件系统实现探索-阿里云开发者社区

开发者社区> KB小秘书> 正文

.Net Micro Framework研究—FAT文件系统实现探索

简介: 由于目前.Net Micro Framework并不支持P/Invoke功能,所以在底层在驱动层面用C直接对存储器(Flash)进行文件系统开发是行不通的。幸好.Net Micro Framework提供了ExtendedWeakReference类,其中赋值Target对象可以把数据存放到存储器上
+关注继续查看

由于目前.Net Micro Framework并不支持P/Invoke功能(也无法像WinCE一样开发流式驱动),所以在底层在驱动层面用C直接对存储器(Flash)进行文件系统开发是行不通的。幸好.Net Micro Framework提供了ExtendedWeakReference类,其中赋值Target对象可以把数据存放到存储器上(Flash)。

[Serializable]
        private class FlashDatas
        {
            //调入数据
            public static byte[] Load(uint index)
            {
                ExtendedWeakReference ewr = ExtendedWeakReference.RecoverOrCreate(
                        typeof(FlashDatas),                       //类型,任意类都可以,其名称起到一个索引作用
                        index,                                    //ID号,这个数据比较有用,不同ID号代表不同数据
                        ExtendedWeakReference.c_SurviveBoot);//该标志和.c_SurviveBoot 区别不大
                return ewr.Target as byte[];
            }        
 
            //保存数据
            public static void Save(uint index, byte[] data)
            {
                ExtendedWeakReference ewr = ExtendedWeakReference.RecoverOrCreate(typeof(FlashDatas), index, ExtendedWeakReference.c_SurviveBoot);
                ewr.Target = data;
            }
    }

上面的代码就是ExtendedWeakReference类的具体使用,从代码可以看出我们无法直接对存储器进行读写,只能通过保存类的方式对数据进行存储,至于该数据存放到何处,那是无从得知的。
我最初的想法是定义一个类,类中定义一个大数组,不过实际调试发现,该数组不能太大,超过几十K就会出现内存溢出。幸好该对象可以是字节数组,所以我产生了另一个想法,每次保存一个512字节大小的字节数组,相当于磁盘上的一个扇区,以此为存取的最小单位,实现FAT文件系统。
在我博客上曾写了一篇关于FAT文件系统的文章,如《FAT文件系统几点释疑》(http://blog.csdn.net/yefanqiu/archive/2008/03/13/2176340.aspx),我们知道要实现FAT16系统一般至少需要4M存储空间,实现FAT32一般需要256M空间以上。所以我简单的做了一个程序,在实际硬件中测试一下存取1024*4个512字节的数组内存是否可行。
测试代码如下:

private const uint SectorSize=512;   //扇区大小
        private const uint SecPerClus = 4;   //一个簇包含的扇区数
        public static void Main()
        { 
            Debug.Print("Start");
            for (uint i = 0; i < 512; i++)    //1024*4
            {
                byte[] bytData = new byte[SectorSize];
                bytData[0] = (byte)(i % 256);
                bytData[bytData.Length - 1] = bytData[0];
                FlashDatas.Save(i, bytData);
               Debug.Print(i.ToString() + " Save " + bytData[0].ToString() + " " + bytData[bytData.Length - 1].ToString());
 
                //byte[] bytData = FlashDatas.Load(i);
                //if (bytData == null)
                //{
                //    Debug.Print(i.ToString() + " Load Error");
                //    break;
                //}
                //else
                //{
                //    Debug.Print(i.ToString() + " Load " + bytData[0].ToString() + " " + bytData[bytData.Length - 1].ToString());
                //}
            }
            Debug.Print("Exit");
 }

让我失望的是,Digi的开发板存储个数一旦超过128个就会出现读失败,新拿来的iPac-9302开发板要好一些,512个之内读写没有什么问题,超过这个数就会出现和Digi开发板一样的问题。需要说明的时,在使用读写的过程中如果不断电,读写都会成功的。一但断电重新读取,读就会失败。(当然在我测试过程中出现了各种各样不同的现象,如只能成功读取前几个)。
杜伟当初还想直接支持FAT32系统呢,目前恐怕FAT16的支持都很困难了,如果实现FAT12系统就有点不值当了。不过杜伟建议说模拟器也支持数据存储功能,所以先在模拟器中实现该功能。
没有想到,模拟器存储器最大存储仅支持1M,开始我还以为我配置参数不当呢,后来反编译了模拟器相关的核心代码,发现1M在代码中就已经写死了,相关内容如下。
反编译 Microsoft.SPOT.Emulator.dll,下面是关键代码

------------------------------------------------------------------------
//内存大小0x10000*16 = 1024*1024 也就是1M空间。
public class FlashManager : MemoryManagerBase
{
    // Fields 存储空间已经写死,就是1M
    private FlashSector[] _flashSectors = new FlashSector[] { 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.Start),
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.End), 
 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.Start), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.Log, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.StorageA, FlashSectorPartition.None),
new FlashSector(0x10000, FlashSectorUsage.StorageA, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.StorageB, FlashSectorPartition.None), 
new FlashSector(0x10000, FlashSectorUsage.StorageB, FlashSectorPartition.End) };
 
....
}
 
//分配内存
internal override void AllocateMemory()
{
        this.ValidateFlashSectorsInternal();
        uint num = 0;
        for (int i = 0; i < this._flashSectors.Length; i++)
        {
            num += this._flashSectors[i].Length;
        }
        base._size = num;
        base.AllocateMemory();    //分配内存 ... ...
        for (int j = 0; j < base._size; j++)
        {
            base._memory[j] = 0xff;
        }
        this.InitializeFlashSectorsInternal();
}
 
//分配内存
internal virtual void AllocateMemory()
{
        this._memory = new byte[this._size];
        this._handle = GCHandle.Alloc(this._memory, 3);
}

此外模拟器在运行结束时,不能保证执行重载的UninitializeComponent函数,所以无法保存内存的数据,代码如下。

/// <summary>
        /// Called by the emulator after all components were setup and registered
        /// </summary>
        public override void InitializeComponent()
        {
            base.InitializeComponent(); 
            _form = new YFEmulatorForm(this.Emulator);
            _form.OnInitializeComponent();
 
            //Launch the UI thread.
            Thread uiThread = new Thread(RunForm);
            uiThread.SetApartmentState(ApartmentState.STA);
            uiThread.Start();
 
            //读Flash数据
            EmulatorFlashPersistance.Load(this);
 
            //必须添加这句,否则不会执行UninitializeComponent方法
            Application.DoEvents();     //这是我添的
        }
        
       
        /// <summary>
        /// Called by the emulator after the program exits
        /// </summary>
        public override void UninitializeComponent()    //这个函数不能保证会运行
        {
            //保存数据
            EmulatorFlashPersistance.Save(this);        //所以无法保存最后的结果
            Application.DoEvents(); //保存数据 
 
            base.UninitializeComponent();
            //When the Micro Framework is shutting down, inform the the WinForm application 
            //to exit as well.
            Application.Exit();
        }

目前该工作的开展对我来说,实在是一个不小的挑战。由于国内研究.Net Micro Framework不多,不仅没有人进行深层次的讨论,也少见相关资料,所以.Net Micro Framework推广真是任重而道远啊。
附记:在我上两篇关于串口部署的文章又有了新的进展,最近花了300多元购买了Moxa的UPort1110 USB转串口设备(一般的杂牌子的设备大约几十元一个)还真不错,在我笔记本上终于可以直接通过串口对.Net Micro Framework进行调试了。 

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
《UNIX环境高级编程(第3版)》——2.3 UNIX系统实现
在McKusick等[1996]的1.1节中给出了UNIX系统家族树的详细历史。UNIX的各种版本和变体都起源于在PDP-11系统上运行的UNIX分时系统第6版(1976年)和第7版(1979年)(通常称为V6和V7)。
1908 0
Android Native禁止使用系统私有库详解
解读Android Native对于系统私有库的限制,老版本的黑科技代码在N版本之后都可能导致APP崩溃。
1661 0
DockOne微信分享(九十七):现有系统实施微服务架构改进经验分享
本文讲的是DockOne微信分享(九十七):现有系统实施微服务架构改进经验分享【编者的话】微服务是最近非常热门的话题了,它带来的好处吸引不少互联网公司对现有项目进行微服务架构改进。 本次分享是博主根据自身的项目经验,介绍如何对现有架构进行调整,总结这过程中的相关技术选型,以及如何实施技改,并分享最终取得的非常让人意外的成果。
1525 0
iOS系统开发提升体验优化方案
随着Flutter等跨端框架的出现,业务开发同学经常需要在Android/IOS上跨端进行业务开发,问题定位等。新的不熟悉的环境的搭建总会遇到各种各样的问题,导致搭建失败,特别是IOS开发环境,是最复杂的,不仅环境搭建繁琐,而且切分支后的打包速度很慢,所以我们设计实现了两个工具,用于优化闲鱼IOS开发体验。
262 0
FAT32文件系统的存储组织结构(二)
<p style="word-wrap: break-word; margin-top: 5px; margin-bottom: 5px; padding-top: 0px; padding-bottom: 0px; color: rgb(102, 102, 102); font-family: 宋体, Arial; font-size: 16px; line-height: 26px;"> 
1162 0
+关注
1090
文章
1253
问答
来源圈子
更多
阿里云最有价值专家,简称 MVP(Most Valuable Professional),是专注于帮助他人充分了解和使用阿里云技术的意见领袖阿里云 MVP 奖项为我们提供了这样一个机会,向杰出的意见领袖表示感谢,更希望通过 MVP 将开发者的声音反映到我们的技术路线图上。
+ 订阅
文章排行榜
最热
最新
相关电子书
更多
文娱运维技术
立即下载
《SaaS模式云原生数据仓库应用场景实践》
立即下载
《看见新力量:二》电子书
立即下载