uClinux下SDRAM数据交换区的生成和使用
引言
数据交换区就是指程序执行中使用到的各种数据所存放的内存空间。本文提出了在uclinux操作统下为嵌入式工业网络服务器开辟数据交换区的思想和解决方案。嵌入式工业网络服务器项目的开发目的是构建一个以client/server模式工作的工业远程监控系统,其前端是一个嵌入式服务器,它安装在工业现场,可以与三菱plc组成的控制系统接口,对该系统实施实时监控。其中数据交换区部分起到承上启下的枢纽作用,向上负责对客户端交互现场设备数据,向下负责与接口缓冲区交互设备实时数据。系统以freescale公司的32位控制芯片cold
fire 5307为cpu,使用两片现代公司的hy57v641620 的4m 16bit的sdram拼成4m 32bit的sdram,使系统具有16m
bytes的ram空间。
uclinux操作系统
系统选取uclinux 操作系统为开发平台,为有两片4m 16bit的sdram的嵌入式工业网络服务器开辟数据交换区。
coldfire 5307微处理器上可以运行很多操作系统,但是uclinux是最合适、性价比最高的操作系统。目前商业实时操作系统,比如vxworks和nuclesus等价格比较昂贵,而且需要附加的c编译器和相关的调试工具。而像ucos等免费的实时操作系统,又没有很好的文件系统和tcp/ip协议的支持,就嵌入式工业网络服务器的应用特点而言,uclinux是最适合的操作系统。我们在uclinux下面实现了jffs(journaling flash file system)文件系统,每次子模块传送回来的数据都可以像日志一样记录下来,非常方便。另外还在主系统中的nand flash接口中实现了fat32文件系统,与pc机上windows下面的fat32文件系统兼容,因此非常方便数据的备份和更新。使用uclinux的同时,主控制系统的所有软件工作都在linux(red hat)系统下使用gcc交叉编译器来编译,使用gdb交叉调试器来调试工作。
uclinux内核采用模块化的设计,很多功能块可以独立地加载或卸载,在设计内核时可以把这些内核模块作为可选项,在编译系统内核时指定。一种较通用的做法是对uclinux内核重新编译,在编译时仔细选择嵌入式系统所需要的功能支持模块,同时删除不需要的功能模块。通过对内核的重新配置,可以使系统运行所需要的内核显著减小,从而缩减需要的控制器ram和rom资源。
uclinux同标准linux的最大区别就在于内存管理,对sdram数据交换区的管理就属于内存管理范畴。标准linux使用虚拟存储器技术,对于uclinux来说,其设计针对没有mmu(memory management unit) 的处理器。但uclinux仍然采用存储器的分页管理,系统在启动时把实际存储器进行分页。在加载应用程序时程序分页加载。但是由于没有mmu管理,所以实际上uclinux采用实存储器管理策略(real memory management)。uclinux系统对于内存的访问是直接的,它对地址的访问不需要经过mmu,而是直接送到地址线上输出,所有程序中访问的地址都是实际的物理地址。 uclinux对内存的管理从编译内核开始,从而系统将在启动的初始化阶段对内存进行分页,并且标记已使用的和未使用的内存。系统将在运行应用时使用这些分页内存。另外由于采用实存储器管理策略,用户程序同内核以及其他用户程序在一个地址空间,程序开发时要保证不侵犯其他程序的地址空间,以使得程序不至于破坏系统的正常工作,或导致其他程序的运行异常。
数据交换区硬件架构
coldfire系列微处理器是freescale公司延续其m68k系列微处理器推出的新一代32位高性能嵌入式微处理器。coldfire系列微处理器继承了m68k系列优秀的指令集设计和cisc架构的优点,融入了risc架构的优点,在速度和架构之间得到了很好的平衡。coldfire 5307微处理器运行在外部总线时钟45mhz、内部总线时钟90 mhz时,可以达到75mips。
与flash存储器相比较,sdram不具有掉电保存数据的特性,但其存取速度大大高于flash存储器,且具有读/写的属性。因此,sdram在系统中主要用于程序的运行空间、数据交换区及堆栈区。当系统启动时,cpu首先从复位地址0x0处读取启动代码,在完成系统初始化后,程序代码一般应调入sdram中运行,以提高系统的运行速度,同时,系统及用户堆栈、运行数据也都放在sdram中。sdram具有单位空间存储容量大和价格便宜的优点,已广泛应用在各种嵌入式系统中。sdram的存储单元可以理解为一个电容,总是倾向于放电,为避免数据丢失,必须定时刷新(充电)。
嵌入式工业网络服务器项目中的数据交换区根据数据交换区配置文件生成,向上负责对客户端交互现场设备数据,向下负责与接口缓冲区交互设备实时数据。每片sdram是由4m 16bit的内部bank组成,这个bank的选择由sdram的外部引脚ba0和ba1来完成,在我们的系统中ba0和ba1都连接在coldfire5307的高端地址线上;其他信号和coldfire5307都一一对应。udqm和ldqm是sdram的高低字节片选信号,因为每片sdram都是16位架构,所以我们将两片sdram拼在一起变成32位时,其他信号线对两片sdram来说都是共用的,唯有数据线和这两个信号不是共用的。这两个信号分别接coldfire5307的cas0、1和cas2、3。sdram与mcf5307的连接关系如图1所示。

数据交换区软件实现
服务器依据数据交换区配置文件生成设备数据交换区。数据交换区格式如表1所示。

数据交换区生成及使用流程图如图2所示。


# include< linux/ boot mem.h>
void*alloc_bootmem(unsigned long size);
void*alloc_bootmem_low(unsigned long size);
void*alloc_bootmem_low-pages(unsignedlong size);
分配内存空间所使用的函数调用如下:
# include
void*malloc(size_t size);
void*calloc(size_t nmemb,size_t size);
函数malloc和calloc都是用于分配动态内存空间的函数。
函数malloc的参数size表示申请分配的内存空间的大小,以字节记。
函数calloc的参数nmemb表示分配的内存空间占的数据项数目。参数size表示每一个数据项的大小,以字节记。也就是说,calloc函数分配大小为nmemb*size大小的内存空间。
calloc函数与malloc函数最大的区别是calloc函数将初始化所分配的内存空间,置所有位置为0。调用成功时,malloc函数与calloc函数的返回值都为被分配的内存空间的指针;调用失败时,返回值为null。动态内存被释放在c中,指针是处理许多数据结构的关键。没有指针,也许根本不能使用动态内存的诸多特性。在编写程序的时候,它允许程序员建立复杂的内存系统,提高了处理未知内容或者类型数据的灵活性。在c中还有一点对字符串处理和数据的输入输出很重要。对指针的彻底了解有助于写出更好、更高效的代码。
如果使用一种算法不够,链表可以解决这个问题。当从未知大小的数据块中读入数据时,用户不得不把数据读到内存中。这是因为处理读入数据的函数,必须把数据读到一块一定大小的内存中。在读入以后,必须找到一种接合分离数据的办法。
一般使用fgets( )把数据读到n个字节大小的内存中。缓冲区是n+1个字节大,但是请注意必须使用1个字节放结束标记。然后使用了一个简单的链表保存数据。这个链表有一个特殊项:一个名为iscontinuing的整型变量。如果这个变量有值,它表示当前结构中的数据不是字符串最后的部分,最后的部分将包含在以后的结构中。这个变量再把数据从内存中调出来,重新组装的时候使用。
还有一种更明智的保存数据的方法。逐步处理数据直到到达数据末尾。首先要修改的是结构的定义。在结构中包含字符串。在结构中定义一个指针,指向动态申请的内存区域。这样做的好处是,字符串可以很长。
现在产生输出的代码更简单了。它做的所有的事情就是计算和显示。不再需要合并字符串,因为它们已经被合并了。
结束语
数据交换区的生成和使用对嵌入式工业网络服务器项目尤为重要,它是web服务器正常运行的先决条件,起到承上启下的枢纽作用。向上负责对客户端交互现场设备数据,向下负责与接口缓冲区交互设备实时数据。日前,该部分设计工作已经完成,并在试运行期间情况良好,达到预期效果。
评论