微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > 嵌入式系统设计讨论 > 墩子学嵌入式-入门之路

墩子学嵌入式-入门之路

时间:10-02 整理:3721RD 点击:

少年易老学难成,   一寸光阴不可轻。

未觉池塘春草梦,   阶前梧叶已秋声。


《劝学》   朱熹


    都说万事开头难,作为一名大学只接触过单片机,且多年未用的选手,我对这个难字深有感触。毕业以来一直都在做整体设计方案,对大学用过的单片机已放下多时,最近由于项目需要,俺要自己去使用arm A9构架的嵌入式板卡。无奈之下只得拿着该开发板的使用手册依葫芦画瓢瞎操作一番,由于手册并没有详细说明指令的意思,自己只能一个个baidu它们,经常出现缺少符号或者字母操作失败的情况。

后来机缘巧合看到了韦东山老师的视频,学了几课以后,顿觉脑子不在一桶浆糊。韦东山老师讲得非常透彻,我激动不已于是一口气下载了视频的全部三期(后来才知道2,3期是需要付费的)。手机里一份,家里电脑里一份,经常在公交车上看看,在家里看看。对于uboot和内核、文件系统这三课看了有三四遍,慢慢的对这三部分有了一些意识和理解,并且学会操作他们。

    原来uboot的最终目的是为了启动内核kernel,但是在启动之前需要完成初始化时钟、关看门狗、初始化串口以及读写flash等操作,具体操作流程如下:

对于uboot的编译操作只需要执行:

      make100ask24x0.config      配置文件

      make                     编译文件

解释:

make 100ask24x0.config

相当于执行@$(MKCONFIG)$(@:_config=) arm arm920t 100ask24x0 NULL s3c24x0

uboot启动流程:

    第一阶段:

板卡上电执行uboot第一个程序start.S,该uboot的入口程序才用汇编编写,代码如下:

/* a, 设置CPU为SVC32模式 */

       mrs  r0,cpsr

       bic  r0,r0,#0x1f

       orr  r0,r0,#0xd3

       msr  cpsr,r0

/* b, 关闭看门狗 */

/* c, 关中断 */

/* d, CPU初始化,初始化SDRAM */

       adr  r0, _start

       ldr  r1, _TEXT_BASE

       cmp      r0,r1

       blne     cpu_init_crit

/* e, 设置stack */

stack_setup:

       ldr  r0, _TEXT_BASE  /* 顶部128 KiB重定位的u-boot */

       sub  r0, r0, #CFG_MALLOC_LEN  /* 向下是内存分配空间 */

       sub  r0, r0, #CFG_GBL_DATA_SIZE /* bdinfo结构体地址空间 */

#ifdef CONFIG_USE_IRQ

       sub r0, r0, #(CONFIG_STACKSIZE_IRQ+CONFIG_STACKSIZE_FIQ)

#endif

       sub sp, r0, #12     /* 预留3个字   */

/* f, 时钟初始化  */      

bl  clock_init      

/* g, 重定位,代码从Flash拷贝到SDRAM中 */

/* h, 清bss段 */

clear_bss:

       ldr  r0, _bss_start      /*  bss段起始地址 */

       ldr  r1, _bss_end        /*  bss段末尾地址  */

       mov  r2, #0x00000000     /* 清零 */

clbss_l:str r2, [r0]      /* bss段地址空间清零循环... */

       add  r0, r0, #4

       cmp  r0, r1

       bne  clbss_l

/* i, 调用start_armboot */

       ldr  pc, _start_armboot

       _start_armboot: .word start_armboot


第二阶段:

    start_armboot是uboot执行的第一个C语言函数,完成系统初始化工作,基本的初始化函数指针保存在init_sequence[]数组中。

    函数void start_armboot (void)

{

            mem_malloc_init(_armboot_start - CONFIG_SYS_MALLOC_LEN);

/*配置Flash */

display_flash_config(flash_init ());

           /* 配置环境变量*/

env_relocate();

......

/*main_loop()循环不断执行 */

for(;;)

{

               main_loop();      /* 主循环函数处理执行用户命令*/

}

}

该程序通过main_loop函数一直等待用户在控制台输入命令,接收到命令后进行命令解析。

启动内核:      

s=getenv(“bootcmd”)

run_command(s)

环境变量中bootcmd参数有一条bootm 0x30007F00指令,表示内核在该地址处启动。

    当有启动命令bootm时,会调用do_bootm函数。这个函数专门用来引导各种操作系统映像,可以支持引导Linux、QNX等操作系统,该函数会调用do_bootm_linux()。

    最终do_bootm_linux()函数会调用

    theKernel(0, bd->bi_arch_number, bd->bi_boot_params)函数来启动内核,传递启动参数。

    uboot就先写这么多吧,好多地方我也不是很清楚,希望以后在分析uboot源码的过程中可以深入理解。

    不知不觉就看完了韦东山老师视频的第一期,回头看看第二期视频的目录,主要讲驱动的,非常吸引人,并且和目前的项目需求比较贴近。于是我迫不及待的购买了第二期视频好好学习一下。

    最后聊聊韦东山老师的讲解风格吧,韦老师边讲解边写代码的风格非常棒,对于嵌入式高手来说些代码也许比较容易,但对于新手来说绝对是授人以渔方式的教学。以前经常在一些帖子里面仅仅只能看到“重要”代码,对于初学者来说这是很无奈的,因为你往往面对着大量代码的移植和修改工作,如无人指导的话,绝对是个噩梦的开始。但现在对于新手来说,咱们有了更好的选择。

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top