uClinux简介

简介: 1. uClinux简介uClinux这个英文单词中u表示Micro,小的意思,C表示Control,控制的意思,所以uClinux就是Micro-Control-Linux,字面上的理解就是"针对微控制领域而设计的Linux系统"。uclinux是一个源码开放的操作系统(http://www.uclinux.org/),面向没有MMU(Memory Management Unit

1. uClinux简介
uClinux这个英文单词中u表示Micro,小的意思,C表示Control,控制的意思,所以uClinux就是Micro-Control-Linux,字面上的理解就是"针对微控制领域而设计的Linux系统"。

uclinux是一个源码开放的操作系统(http://www.uclinux.org/),面向没有MMU(Memory Management Unit)的硬件平台。它是linux的一个变种,主要的区别在于两者的内存管理机制和进程调度管理机制,同时为了适应嵌入式应用的需求,它的采用了romfs文件系统,并对linux上的c语言库glibc做了简化。

2. 硬件体系结构简介

运行uClinux的硬件平台主要包括如下几个部分:cpu(ARMv4指令集兼容)、uart、memory controller、定时器、flash存储器,sdram存储器,中断控制器和DMA。

3. 编译环境和编译工具。

uclinux操作系统源码绝大部分是用c语言开发的,有一些与硬件直接相关的代码则用特定于某一CPU体系结构的汇编来实现。这些源码只能用GNU的gcc编译工具来进行编译、链接。

GNU gcc可以运行于Linux/Unix操作系统上。如果要在Windows平台上运行gcc,则必须安装Cygwin。Cygwin可以在Windows中安装一个linux的运行环境,这样就可以在windows下运行原本只能在linux下运行的程序。

为了在PC上编译得到运行于目标CPU上的操作系统内核,还必须安装一个合适的交叉编译器。Gcc 提供了现成的针对MIPS、ARM、M68K、Sharc、PowerPC的交叉编译器。如果没有现成的交叉编译器,则需要自行设计。GNU网站提供了一些如何开发新的交叉编译器的文章。开发一个新的编译器,一般需要如下几个步骤:

1、编写机器描述脚本。采用gcc的RTL(Register Tansfer Language)语言描述针对某一CPU体系结构的机器指令与寻址方式、CPU浮点处理方式、endianess、c语言中各种数据类型的位宽、寄存器的个数和使用规则、堆栈和函数调用规则等体系结构的细节。
2、设计代码生成器。Gcc在对c语言源文件进行了词法和语法分析后,将产生一种中间格式文件(intermediate representation)。为了把这种中间格式文件转化为针对具体CPU体系结构的机器码,需要自行设计一个代码生成器。
3、设计汇编器
4、设计链接器


4. uClinux启动过程

uClinux系统的启动可以分为两个步骤:
1. 运行bootloader初始化程序

SRAM、SDRAM等存储设备属于挥发性的存储器,掉电以后其中的内容就会全部丢失,所以必须把操作系统的内核镜像存放在Flash等不挥发性存储介质上。但是操作系统在运行时,需要动态的创建一些如数据段、堆栈、页表(针对使用虚拟地址的操作系统)等内容,所以需要在RAM中运行操作系统。因此,就需要一个引导程序把操作系统的内核镜像从Flash存储器拷贝到RAM中,然后再从RAM中执行操作系统的内核。Bootloader就是可以完成这样一种功能的程序。

从本质上来讲,bootloader不属于操作系统内核。它采用汇编语言编写,因此针对不同的cpu体系结构,这一部分代码不具有可移植性。在移植操作系统时,这部分代码必须加以改写。

具体来讲,bootloader在系统启动时主要完成以下几项工作:
(1) 将操作系统内核从flash拷贝到sdram中,如果是压缩格式的内核,还要将之解压缩。
(2) 改写系统的memory map,原先flash起始地址映射为0地址,这时需要将RAM的起始地址映射为0。
(3) 设置堆栈指针并将bss段清零。将来执行c语言程序和调用子函数时要用到。
(4) 改变pc值,使得cpu开始执行真正的操作系统内核。
2. 运行操作系统内核
bootloader程序执行完上述的各项工作后,通过一条跳转指令,转而执行ini目录下c语言源文件main.c中的函数start_kernel()。因为在此之前bootloader已经创建好一个初始化环境,
c函数可以开始执行了。整个操作系统内核的初始化工作从这里才算是真正开始。这个函数的长度比较短


代码如下:

 

asmlinkage  void  __init start_kernel( void )
{
 
char   *  command_line;
 unsigned 
long  mempages;
 
extern   char  saved_command_line[];


/*
 * Interrupts are still disabled. Do necessary setups, then
 * enable them
 
*/
 lock_kernel();
 printk(linux_banner);
 setup_arch(
& command_line);
 printk(
" Kernel command line: %s  " , saved_command_line);
 parse_options(command_line);
 trap_init();
 init_IRQ();
 sched_init();
 softirq_init();
 time_init();

 
/*
 * HACK ALERT! This is early. We're enabling the console before
 * we've done PCI setups etc, and console_init() must be aware of
 * this. But we do want output early, in case something goes wrong.
 
*/
 console_init();
#ifdef CONFIG_MODULES
 init_modules();

#endif
 
if  (prof_shift) {
 unsigned 
int  size;
 
/*  only text is profiled  */
 prof_len 
=  (unsigned  long & _etext  -  (unsigned  long & _stext;
 prof_len 
>>=  prof_shift;
 
 size 
=  prof_len  *   sizeof (unsigned  int +  PAGE_SIZE - 1 ;
 prof_buffer 
=  (unsigned  int   * ) alloc_bootmem(size);
 }

 kmem_cache_init();
 sti();
 calibrate_delay();
#ifdef CONFIG_BLK_DEV_INITRD
 
if  (initrd_start  &&   ! initrd_below_start_ok  &&
 initrd_start 
<  min_low_pfn  <<  PAGE_SHIFT) {
 printk(KERN_CRIT 
" initrd overwritten (0x%08lx < 0x%08lx) -  "
 
" disabling it.  " ,initrd_start,min_low_pfn  <<  PAGE_SHIFT);
 initrd_start 
=   0 ;
 }

#endif
 mem_init();
 kmem_cache_sizes_init();
 pgtable_cache_init();

 mempages 
=  num_physpages;

 fork_init(mempages);
 proc_caches_init();
 vfs_caches_init(mempages);
 buffer_init(mempages);
 page_cache_init(mempages);

#if  defined(CONFIG_ARCH_S390)
 ccwcache_init();

#endif
 signals_init();
#ifdef CONFIG_PROC_FS
 proc_root_init();

#endif

#if  defined(CONFIG_SYSVIPC)
 ipc_init();

#endif
 check_bugs();
 printk(
" POSIX conformance testing by UNIFIX  " );

 
/*  
 * We count on the initial thread going ok 
 * Like idlers init is an unlocked kernel thread, which will
 * make syscalls (and thus be locked).
 
*/
 smp_init();
 rest_init();
}

5. 系统源码的修改

移植一个操作系统到新的硬件平台,比较好的办法是寻找一个架构相近并且已经做好操作系统移植的硬件平台。然后,对原先的操作系统做一定修改。系统源码修改的工作量取决于自行设计的硬件平台与现有的硬件平台之间差异程度。此设计中的硬件平台与三星4510芯片较为接近,并且也可以下载到针对4510b的uclinux系统源码。所以可以从此源码入手,根据我们的硬件平台与4510b的不同之处,在源码中找到相应的文件并加以修改。下面介绍如何修改系统源码。


需要修改的系统源码主要有如下几处:
(1) bootloader相关代码。此代码位于/uClinux/linux-2.4.x/arch/armnommu/boot/compressed/目录下名为head.s的文件中。此处程序用汇编语言实现,需要修改的地方主要是设置memory map的代码,与memory controller的硬件实现相关。
(2) UART相关代码。UART相关代码位于/uClinux/linux-2.4.x/drivers/char目录下的serial.c
(3) 定时器相关代码。uClinux中有如下函数调用star_kernel()->time_init()->setup_timer(),需要修改setup_timer()函数中的相关代码。
(4) 中断控制器相关。/uClinux/linux-2.4.x/arch/armnommu/irq.c

除了上述的代码,还有其他多处需要修改。在修改源代码时,可按照uclinux启动和执行顺序依次修改整个平台。熟悉linux内核源码结构对操作系统移植有很大帮助。
 

相关文章
|
6月前
|
算法 程序员 编译器
【C/C++】C/C++编程——C/C++简介
【C/C++】C/C++编程——C/C++简介
49 0
|
6月前
|
编译器 C语言 C++
PCLINT 简介
PCLINT 简介
115 0
|
监控 Oracle 关系型数据库
oratop的使用简介
oratop是采用类似top命令的方式实时监控oracle数据库,包括rac和non-rac数据库
115 0
|
运维 Kubernetes Ubuntu
Kebernetes简介
Kebernetes简介
231 0
|
XML 分布式计算 算法
Rosetta | Rosetta简介
Rosetta | Rosetta简介
908 0
Rosetta | Rosetta简介
|
数据安全/隐私保护
叶帆密码箱简介
叶帆密码箱一个纯绿色软件,小巧实用,可以保存各种网络账号,银号帐户,Email信息 等等,此外还可以直接打开链接网址,直接通过热键发送登录信息,方便简洁
733 0
|
JavaScript 前端开发 C++
cheerp 简介
这个文章主要介绍了为什么要用cheerp技术以及和其他对标的技术对比,他的闪光点在哪里。 WebAssembly 是一种中间码用于加速浏览器端应用,目前有多种语言可以编译或者交叉到这种格式。当然这个不是本文的内容,详细了解请移步 wasm官网 目前支持: cc++是官方推荐的方式,详细使用见文档;其他语言 AssemblyScript:语法和 TypeScript 一致,对前端来说学习成本低,为前端编写 WebAssembly 最佳选择; Rust:语法复杂、学习成本高,对前端来说可能会不适应。
2703 0
|
C#
C#中的NameValueCollection简介
NameValueCollection继承自NameObjectCollectionBase,并且和一般的键值对不同的是,它支持集合中出现相同的Key。 引用:using System.Collections.
1761 0
|
缓存 Shell 开发工具
antigen简介
在Linux下经常需要进行终端操作,一个好的shell解释器可以帮助我们极大提高工作效率。有些同学可能听说过zshell和oh-my-zsh,前者是一个shell程序,由于功能比较强大,所以叫做z shell,意为终极shell;而后者是一个模板化的zshell配置脚本,通过它我们可以快速配置一个好用的zsh。
1581 0
|
Web App开发 存储 安全