您好、欢迎来到现金彩票网!
当前位置:大赢家 > 复位序列 >

嵌入式软件开发之: 复位和初始化

发布时间:2019-04-25 12:18 来源:未知 编辑:admin

  处理程序是用汇编语言编写的代码块,它在系统复位时执行,完成系统初始化操作。对于具有局部存储器的内核,如Caches、紧密藕荷存储器(TCM)、存储管理单元(MMU)和存储器保护单元(MPU)等,在初始化过程这一阶段完成必要的配置。复位处理程序在执行之后,通常跳转到__main以开始C库的初始化序列。

  所有的ARM系统都有一个向量表(vector table)。向量表不是初始化序列的一部分,但是对每个要处理的异常,它必须存在。这些地址通常包含以下形式的跳转指令。

   LDR pc,[pc,offset]:这条指令将异常处理程序的入口地址从存储器装载到pc。该地址是一个32位的绝对地址。由于有额外的存储器访问,装载跳转地址会使分支跳转到特定处理程序,给系统执行带来延时。不过,可以使用这种方法跳转到存储空间内的任意地址。

   MOV pc,#immediate:将一个立即数复制到pc。使用该指令可以跨越整个地址空间,但是受到地址对齐问题的限制。这个地址必须由8位立即数循环右移偶数次得到。

  另外,也可以在向量表中使用其他类型的指令。例如,FIQ处理程序可以从地址0x1c处开始执行。因为它位于向量表的最后,这样FIQ处理程序就可以不用跳转,立即从FIQ向量地址处开始执行。

  在向量表的入口处要有ENTRY标识。该标识通知链接程序该代码是一个可能的入口点,因而在链接时,不能被清除。

  启动时,0x0处必须要有一条有效指令,因此,复位时0x0000地址必须为非易失性存储器,如ROM或FLASH。

  注意有些系统是从0xffff0000处开始执行的,对于这样的系统,地址0xffff0000处必须为非易失性存储器。

  可以将ROM定位在0x0处。但是,这样配置有几个缺点。首先ROM存取速度通常较RAM要慢,当跳转到异常处理程序时,系统性能可能会大受影响。其次,将向量表放于ROM中,运行时不能修改。

  存储器地址重映射(Memory Remap)是当前很多先进控制器所具有的功能。所谓地址重映射就是可以通过软件配置来改变存储器物理地址的一种机制或方法。

  当一段程序对运行自己得存储器进行重映射时,需要特别注意保证程序执行流程在重映射前后的承接关系。实现重映射的关键就是要使程序指针在remap以后能继续往下得到正确的指令。本书中介绍两种实现重映射的机制,不同的系统可能会有多种灵活的remap方案,用户在具体实现时要具体分析。

  原来RAM和ROM各有自己的地址,进行重映射以后RAM和ROM的地址都发生了变化。这种情况下,可以采用以下方案。

  ④ 因为RAM在remap前准备好了内容,使得PC指针能继续在RAM里取到正确的指令。

  系统上电后的缺省状态是0x0地址上放有ROM。这块ROM有两个地址:从0起始和从0x10000起始,里面存储了初始化代码。当进行地址remap以后,从0x0起始的地址被定向到RAM上,ROM则只保留有惟一的从0x10000起始的地址。

  如果存储在ROM里的复位异常处理程序(Reset-Handler)一直在0x0~0x4000的地址上运行,则当执行完remap以后,下面的指令将从RAM里预取,这必然会导致程序执行流程的中断。根据系统特点,可以用下面的办法来解决这个问题。

  ① 上电后系统从0x0地址开始自动执行,设计跳转指令在remap发生前使PC指针指向0x10000开始的ROM地址中去,因为不同地址指向的是同一块ROM,所有程序能够顺利执行。

  ② 这时候0x0~0x4000的地址空间空闲,不被程序引用,执行remap后把RAM引进。因为程序一直在0x10000起始的ROM空间里运行,remap对运行流程没有任何影响。

  ③ 通过在ROM里运行的程序,对RAM进行相应的代码和数据拷贝,完成应用程序运行的初始化。

  该ROM与RAM地址重映射的方法可以应用于任何具有ROM/RAM重映射机制的平台,但是内存重映射的地址根据具体平台的不同而不同。

  图13.16显示的地址重映射例子中,第一条指令实现从ROM临时地址(0x0地址)到实际ROM的跳转。然后,控制寄存器的重映射位,清除ROM的临时地址设置。该代码通常在系统复位后立即执行。重新映射必须在执行C库初始化代码前完成。

  下面的例子显示了在ARM的Integrator开发板上实现的ROM/RAM重映射过程。

  许多ARM处理器内核具有片上存储器系统,如MMU或MPU。这些设备通常是在系统启动过程中进行设置并启用的。因此,带有局部存储器系统的内核的初始化序列需要特别地考虑。

  在前面所述的代码启动的过程中,__main中C库初始化代码负责建立代码执行时的内存映像,在跳转到__main前,必须建立处理器内核的运行时存储器视图。这就是说,在复位处理程序中必须设置并启用MMU或MPU。

  另外,在跳转到__main前(通常在MMU/MPU设置前),必须启用紧耦合存储器TCM(Tightly coupled Memory),因为在通常情况下都是采用分散加载方法将代码和数据装入TCM。当TCM启用后,用户不必存取由TCM屏蔽的存储器。

  在跳转到__main前,如果启用了Cache,可能还会遇到Cache一致性的问题,__main中的函数将程序代码从其加载域拷贝到执行域,在此过程中将指令作为数据进行处理。这样,一些指令可能被放入数据Cache中,在执行这些指令时,由于找不到地址路径而产生错误。为了避免Cache一致性的问题,在C库初始化序列执行完成后再启用Cache。

  在程序的初始化代码中,用户必须要为处理器用到的各种模式设置堆栈,也就是说,复位处理程序必须为应用程序所使用的任何执行模式的栈指针分配初始值。

  为了设置栈指针,进入每种模式(中断禁用)并为栈指针分配适合的值。要利用软件栈检查,也必须在此设置栈限制。

  下面的例子显示了如何实现__user_initial_stackheap(),该段代码可以和上面的堆栈指针设置程序配合使用。

  一般情况下,系统初始化代码和主应用程序是分开的。系统初始化要在主应用程序启动前完成。但部分与硬件相关的系统初始化过程,如启用Cache和中断,必须在C库初始化代码执行完成后才能执行。

  为了在进入主应用程序之前,完成系统初始化,可以使用$sub和$super函数标识符在进入主程序之前插入一个例程。这一机制可以在不改变源代码的情况下扩展函数的功能。

  下面的例子说明了如何使用$sub和$super函数标识。链接程序通过调用$sub$$main()函数取代对main()的调用。所以用户可以在自己编写的$sub$$main()例程中启用Cache或使能中断。

  在完成硬件初始化之后,必须考虑主应用程序运行在何种模式。如果应用程序运行在特权模式(Privileged mode),只需在退出复位处理程序前切换到适当的模式;如果应用程序运行在用户模式下,要在完成系统初始化之后,再切换到用户模式。模式的切换工作,一般在$Sub$$main(void)函数中完成。

http://ccsagresso.com/fuweixulie/43.html
锟斤拷锟斤拷锟斤拷QQ微锟斤拷锟斤拷锟斤拷锟斤拷锟斤拷锟斤拷微锟斤拷
关于我们|联系我们|版权声明|网站地图|
Copyright © 2002-2019 现金彩票 版权所有