1、达内嵌入式南宁达内嵌入式实时 Linux 的技术研究PC 硬件使用的增加是近年来高端嵌入式系统的一个最重要的发展。由于这个趋势,嵌入式系统的硬件成本大大地下降了,然而应用于嵌入式 PC 平台的软件系统却没有太多的选择。目前嵌入式操作系统主要有 Palm OS,WindowsCE,EPOC,LinuxCE,QNXECOS,LYNX等,但这些系统一般都价格昂贵、不具有良好的可移植性。由于 linux 具有适应于多种CPU 和多种硬件平台、性能稳定、裁剪性能好,开发和使用都很容易等特点,越来越多的人倾向于将 Linux 嵌入到移动计算平台、信息家电、媒体手机及其它产品中去。这同时也对 Linux 的
2、实时性提出了更高的要求。我们参加开发的上海数字技术中心的电子警察系统就是这样一个基于 PC 平台的嵌入式系统。它的主要功能是安装在十字路口上监视闯红灯的汽车,录制这些镜头,并适时通过网络传回总监控中心。该系统原来采用 Vxworks 作为运作平台,然而由于 Vxworks 的昂贵,且只能安装一台机器,因此将 Linux 改制成嵌入式的、具有一定实时性的系统平台。1 硬件需求考虑到实际的系统功能和系统开发的难度,我们决定选择 X86体系作为平台,底板采PCM-5864/L 板,CPU 选择 Intel 的 P55C 系列,同时它也支持 PC104接口。I/O 方面,PCM-5864/L 支持 E
3、IDE、FDD、Keyboard、Mouse、RS-232、USB 等接口,同时也集成了10Mbps/l00Mbps 网卡。另外 PCM-5864/L 还集成了显卡,并提供36位 TTL 的 LCD 接口、LVDS 接口和 Video-in/TV-out 接口。由于电子磁盘体积小、抗震性能也较好的特点,我们选用 M-Systems 公司的 DiskOnChip2000作为存储设备,这样比较适合该系统户外作业的特性。Doc2000还提供了安装的工具包、Linux 下的驱动程序和对不同版本 Linux 内核的Patch。安装时只需对 Linux 内核和 Lilo 进行相应修改即可。这样的硬件选择使
4、系统开发的难度大大降低了。2 嵌入 Linux 主要解决的几个问题由于采用 X86体系作为平台,大大降低了嵌入化 Linux 的难度,主要要考虑的问题可分为以下几个方面:达内嵌入式2.1 非虚拟内存Linux 采用虚拟内存技术,当数据溢出内存时,可以将其交换到磁盘交换空间巾去,这对程序员来说是不可见的。然而,普通的嵌入式系统不需要这种强大的功能。实际上,可能不希望它在实时的关键系统里,因为它会带来无法控制的时问因素。因此,考虑将虚拟内存的功能去掉。然而,清除 Linux 的虚拟内存代码非常费事,我们采用一种迂回的方法,即将交换空间的大小设置为零。这样,如果你写的程序比实际的内存大,系统就会当作
5、你的运行用尽了交换空间来处理;这个程序将不会运行,或者 malloc 将会失灵。这只是一种临时的解决方案,系统中存在许多冗余未清除的代码,如果进一步改进可以缩小系统体积。2.2 处理文件系统许多嵌入式系统没有磁盘或者文件系统,Linux 不需要它们也能运行,这样可以在系统启动时就将操作系统和预先编好的应用程序全都加载到内存中去。然而考虑到以后的扩展,我们分析了 Linux 当前的文件系统,将与系统和程序运行相关的文件析取出来,移植到 Doc 上面这样就能为应用程序的开发人员提供更方便的接口。2.3 设置启动(Lilo 和 BIOS)当 PC 系统启动时,由 BIOS 执行了一些低水平的 CPU
6、 初始化和其它硬件的配置,然后辨认哪个磁盘里有操作系统,把操作系统复制到 RAM 并且转向它。在 PC 上运行的 Linux 依靠 PC 的 BIOS 来提供这些配置和 OS 加戟功能。由于选择了 X86系列作为平台,同样可以通过设置 BIOS 来设定由 Doc 启动。我们将内核安装在 Doc 上。同时修改目录/etc 下的lilo.config 文件,使 Lilo 记录内核代码在存储设备上的位置,然后将 Lilo 安装在 Doc上。这样当系统启动时BIOS 指定 Doc 为启动设备,这时 Lilo 被执行。它指出 Linux 内核的位置,加载内核。2.4 设备驱动程序的编写由于采用通用 PC
7、 平台,有许多驱动程序都不需要自己再编写了。系统中,唯一需要编写驱动程序的设备是上海数字中心的数据采集卡该卡采用的是 Bt848芯片。Bt848是一块很常用的芯片许多驱动程序都非常相近,因此编写它的驱动程序有很好的例子可以参考,达内嵌入式相应的,编写应用程序只需调用驱动程序提供的接口即可,如同操作文件一样。3 Linux 的实时性分析以上简要介绍了实现 Linux 的嵌入化过程,在嵌入化 Linux 的同时,还考虑提高它的实时性。于是着手分析 Linux 的实时性能和存在的不足。概括来讲,影响操作系统实时性能的主要有3个方面:(1)外部中断管理我们知道,外部中断发生时,操作系统调用中断处理程序
8、进入核心态。为了保证系统执行的正确性,要求内核状态不重入,也即保证这部分关键代码执行结束之前不被打断。因此,这时进入关中模式,这是外部中断管理中影响 Linux 性能的一个关键的地方。在这段时间内,操作系统负责将中断发送到相应的设备驱动程序去处理,系统不能进行其它任何工作,为了减少这个过程损耗的时间,Linux 内核利用底半处理过程(bottom-half-handler)帮助实现中断的快速处理。在 Linux 设备驱动程序中,往往将最关键最迅速的部分处理完成之后,将剩余部分任务放置到队列中。当中断响应完成后再执行剩余部分的任务。在 Linux 中,主要设置了以下几个数据结构来标志未完成的任务
9、。enumTIMER_BH=0,CONSOLE_BH,TQUEUE_BH,DIGI_BH,SERIAL_BH,RISCOM8_BH,SPECIALIX_BH,ESP_BH,NET_BH,SCSI_BH,IMMEDIATE_BH,KEYBOARD_BH,CYCLADES_BH,CM206_BH,JS_BH,MACSERIAL_BHISICOM_BH;上面每一项标识未完成任务的队列类型,不同队列的任务轻重缓急不同。extern unsigned long bh_active;extern unsigned long bh_mask;extern void(*bh_base32)(void);bh_
10、base 代表的指针数组中可包含32个不同的底半处理过程。bh_mask 和札 bh_active 的数据位分别代表对应的底半处理过程是否安装和激活。如果 bh_mask 的第 N 位为1则说明bh_base 数组的第 N 个元素包含某个底半处理过程的地址;如果 bh_active 的第 N 位为1则说明必须由调度程序在适当的时候调用第 N 个底半处理过程。这些数据结构的设置一般是在外部设备初始化和中断处理函数运行时进行的。如:在 serialc 中进行 serial 设备达内嵌入式的初始化,它调用语句 init_bh(SERIAL_BH,do_serial_bh);来设置 bh_base数组
11、中相应于 SERIAL_BH 的那一项。又如:在 serial 设备的处理程序中通过语句 queue_task(将不是很紧急的任务放入 tq_serial 队列中,等中断处理函数结束,由 bh_base 中注册的底半处理程序处理队列中的任务。中断管理的第二个关键部分即是系统是否允许中断嵌套的能力,也就是说,当响应一个中断时,是否允许其它更高优先级的中断打断,等更高优先级的中断处理完毕,是否还能恢复原来中断处理的现场。通过这项功能,系统设计者可以指示外部中断的优先级,从而确保高优先级的任务能及时处理。Linux 允许中断嵌套,它是利用外部中断管理器来设置中断的优先级的。在 Linux 的中断处理
12、程序的启动过程中,它一般调用语句mask_and_ack_8259A(irq);来设置8259中的 int_mask 寄存器使优先级比此中断低的中断不能发生。在中断处理程序离开时,调用 enable_8259A_irq(irq)来改回8259中int_mask 寄存器原来的值。因此,intr 类中断的优先级由硬件8259来决定由此可见,Linux 的中断管理部分具有高效的特点,已经可以满足许多软实时任务的要求。(2)进程抢先调度在许多控制系统中,实时控制软件是非常简单的,可以直接写入中断处理程序中与一个特定的中断联系起来。还有一些就不那么简单了,必须开启专门的用户进程为它服务。这时当这个高优先
13、级的进程提交时,如有其它进程正在运行,它就必须打断正在运行的进程。若正在运行的进程运行在用户态,系统一般允许它被打断且执行其它优先权高的进程,若正在运行的进程运行在系统态,则此时是否允许被打断决定了系统是抢先式的还是非抢先式的。Linux 就是一个非抢先式的操作系统,在用户执行系统调用时,不允许其它进程的调度,这样就影响了系统的响应度。一个真正的抢先式的操作系统允许正在系统状态下的当前进程被打断,然后进程切换回来时还能继续从刚才的执行点继续下去。但某些关键部分的代码段。系统必须保证其原子性,并防止重入。通常有如下几种方法:达内嵌入式在关键代码断前关闭中断,等其执行完毕之后再将中断打开; 设计一
14、个信号量在关键代码段之前加锁,在其后解锁; 在系统代码中保证安全的地方加入切换进程的代码 switch(),防止该进程长久占用 CPU,允许调度其它进程; 在关键代码段加入一个 switchaccept 标志,开始该代码段时。将此标志置为否离开时再置回原来的值这样在执行该段代码时,即使进程调度器被激活,它也会先检查此标志。若为否,则返回,并不进行进程切换。 (3)进程调度策略第三个影响系统响应速度的关键部分就是进程调度的策略。对于一个实时性能强的操作系统来说,系统必须规定不同进程的优先级,并把优先级作为唯一的进程选择的标准。Linux 的后期版本参照 Posixl.b 标准,在某些方面已经具备
15、了一些实时操作系统的特性。Linux 有两种类型的进程:一般进程和实时进程,它可以通过 sched_setscheduler 系统调用设置实时进程。实时进程比所有一般进程的优先级高,Linux 设置实对进程的权重为它的 counter 值加1000;设置一般进程的权重为 counter。因此,实时进程总会被认为是最值得运行的进程。然而,Linux 核心的设计主要集中在应用程序的吞吐量上。追求吞吐量的必然结果,就是 Linux 调度器运用一种“公平共享“的策略保证所有的进程得到平均的 CPU 资源。而且,Linux 的进程调度器只是简单地将标有实时标志的进程的权重加1000,至于实时进程间的轻重
16、缓急还没有周密的完整的设计。因此,Linux 的进程调度器还远不能称作是一个真正的实时进程凋度器。4 拟采用的策略根据以上分析的特点,我们决定主要从以下4个方面来修改 Linux 的核心代码。(1)在内核中插入抢先点 由于 Linux 是一个非抢先式的操作系统。因此当一个实时进程提交时,很可能因为当前的进程正处于核心态不能被打断而不能得到及时的处理。因此有必要在 Linux 内核中插入抢先点,使实时进程得到处理。根据上一节分析的特点,太体有4种方法可供选择。权衡这4种方法的利弊,我们决定采用第4种方法,即在关键代码段加入一个 switchaccept 标志,开始该代码段时,将此标志置为否离开时
17、再置回原来的值。达内嵌入式这种方法比采甩 semaphore 的好处是,如果采用许多种 semaphore 的话要考虑是否会产生死锁的问题。比采用锁中断的好处是将中断锁住将丢失中断,而这样不会。而以固定的周期加 switch 语句显然有失灵活性。这样采用这种方法,需要我们分析 Linux 所有系统调用的代码,画出其结构流程图。分析出哪些部分是关键部分,也即不允许置入的部分。在关键代码前后更改 switchaccept 标志。这项工作比较艰巨。同时修改进程调度器,使其判断 switchaccept 标志来决定是否执行进程切换。(2)修改进程调度器 Linux 的进程调度器虽然已经具有一定的实时性
18、能,但还远远达不到真正实时调度器的标准,因此需要修改其进程调度器,必要的话可让 Linux 运行在两种模式下,实时模式和分时模式。可设计一些相关的系统调用,并在实时进程提交时,将系统转化为实时模式,当实时进程结束时,再转化为分时模式。(3)Linux 的中断管理根据前面分析过的,Linux 的中断管理及时地将紧要的任务完成后,将其余不重要的缓慢的任务放置在任务队列中,等到系统空闲(cpu idle()或系统调用等返回时再完成这些任务,这样就提高了系统的响应速度,同时,Linux 还支持中断嵌套。因此,不再对其作太大改动。(4)锁定内存 在本项目的规划中本打算实现 Linux 锁内存的功能,使优先权高的进程在内存中的数据不被换出,从而提高实时进程的运行速度。然而,在分析了 Linux 代码后,发现后来版本的 Linux 已通过系统调用 sys mlock 实现了此项功能。5 结束语采用上述方法修改了内核代码后,由于每个修改方案都是有一定的代价的,它在增加了系统响应速度的同时也在某种程度上降低了系统的整体效率,比如说将内核设置成可抢先的,在进程频繁的切换过程中也要消耗一定的 cpu 处理时间。因此,还需要对各种解决方案进行测试、比较。另外,为了减少嵌入式 Linux 自身的长度,在存储管理部分对虚拟内存也应作进一步的处理。