转自:http://www.cnblogs.com/armlinux/archive/2011/11/06/2396787.html

1.       内核启动地址
1.1.   名词解释
ZTEXTADDR

解压代码运行的开始地址。没有物理地址和虚拟地址之分,因为此时MMU处于关闭状态。这个地址不一定时RAM的地址,可以是支持读写寻址的flash等存储中介。

Start address of decompressor. here's no point in talking about virtual or physical addresses here, since the MMU will be off at the time when you call the decompressor code. You normally call   the kernel at this address to start it booting. This doesn't have to be located in RAM, it can be in flash or other read-only or      read-write addressable medium.

ZRELADDR

内核启动在RAM中的地址。压缩的内核映像被解压到这个地址,然后执行。

This is the address where the decompressed kernel will be written, and eventually executed. The following constraint must be valid:

__virt_to_phys(TEXTADDR) == ZRELADDR

The initial part of the kernel is carefully coded to be position independent.

TEXTADDR

内核启动的虚拟地址,与ZRELADDR相对应。一般内核启动的虚拟地址为RAM的第一个bank地址加上0x8000。

TEXTADDR = PAGE_OFFSET + TEXTOFFST

Virtual start address of kernel, normally PAGE_OFFSET + 0x8000.This is where the kernel image ends up. With the latest kernels, it must be located at 32768 bytes into a 128MB region. Previous kernels placed a restriction of 256MB here.

TEXTOFFSET

内核偏移地址。在arch/arm/makefile中设定。

PHYS_OFFSET

RAM第一个bank的物理起始地址。

Physical start address of the first bank of RAM.

PAGE_OFFSET

RAM第一个bank的虚拟起始地址。

Virtual start address of the first bank of RAM. During the kernel

boot phase, virtual address PAGE_OFFSET will be mapped to physical

address PHYS_OFFSET, along with any other mappings you supply.

This should be the same value as TASK_SIZE.

1.2.   内核启动地址确定
内核启动引导地址由bootp.lds决定。 Bootp.lds : arch/arm/bootp

OUTPUT_ARCH(arm)

ENTRY(_start)

SECTIONS

{

. = 0;

.text : {

_stext = .;

*(.start)

*(.text)

initrd_size = initrd_end - initrd_start;

_etext = .;

}

}

由上 .= 0可以确定解压代码运行的开始地址在0x0的位置。ZTEXTADDR的值决定了这个值得选取。

Makefile : arch/arm/boot/compressed

如果设定内核从ROM中启动的话,可以在make menuconfig 的配置界面中设置解压代码的起始地址,否则解压代码的起始地址为0x0。实际上,默认从ROM启动时,解压代码的起始地址也是0x0。

feq ($(CONFIG_ZBOOT_ROM),y)

ZTEXTADDR := $(CONFIG_ZBOOT_ROM_TEXT)

ZBSSADDR    := $(CONFIG_ZBOOT_ROM_BSS)

else

ZTEXTADDR :=0                                                       ZBSSADDR := ALIGN(4)

endif

SEDFLAGS    = s/TEXT_START/(ZTEXTADDR)/;s/BSSSTART/(ZTEXTADDR)/;s/BSSSTART/(ZBSSADDR)/

……

(obj)/vmlinux.lds:(obj)/vmlinux.lds:(obj)/vmlinux.lds.in arch/arm/mach-s3c2410/Makefile .config

@sed "(SEDFLAGS)"<(SEDFLAGS)"<< > $@

@sed "(SEDFLAGS)"<(SEDFLAGS)"<< > $@ 规则将TEXT_START设定为ZTEXTADDR。TEXT_START在arch/arm/boot/compressed/vmlinux.lds.in 中被用来设定解压代码的起始地址。

OUTPUT_ARCH(arm)

ENTRY(_start)

SECTIONS

{

. = TEXT_START;

_text = .;

.text : {

_start = .;

*(.start)

*(.text)

*(.text.*)

……

}

}

内核的编译依靠vmlinux.lds,vmlinux.lds由vmlinux.lds.s 生成。从下面代码可以看出内核启动的虚拟地址被设置为PAGE_OFFSET + TEXT_OFFSET,而内核启动的物理地址ZRELADDR在arch/arm/boot/Makefile中设定。

OUTPUT_ARCH(arm)

ENTRY(stext)

SECTIONS

{

#ifdef CONFIG_XIP_KERNEL

. = XIP_VIRT_ADDR(CONFIG_XIP_PHYS_ADDR);

#else

. = PAGE_OFFSET + TEXT_OFFSET;

#endif

.init : {                  /* Init code and data             */

_stext = .;

_sinittext = .;

*(.init.text)

_einittext = .;

……

}

}

# arch/arm/boot/Makefile

# Note: the following conditions must always be true:

#   ZRELADDR == virt_to_phys(PAGE_OFFSET + TEXT_OFFSET)

#   PARAMS_PHYS must be within 4MB of ZRELADDR

#   INITRD_PHYS must be in RAM

ZRELADDR    := $(zreladdr-y)

#---> zrealaddr-y is specified with 0x30008000 in arch/arm/boot/makefile.boot

PARAMS_PHYS := $(params_phys-y)

INITRD_PHYS := $(initrd_phys-y)

export ZRELADDR INITRD_PHYS PARAMS_PHYS

通过下面的命令编译内核映像,由参数-a, -e设置其入口地址为ZRELADDR,此值在上面ZRELADDR    := $(zreladdr-y)指定。

quiet_cmd_uimage= UIMAGE $@

cmd_uimage = (CONFIGSHELL)(CONFIGSHELL)(MKIMAGE) -A arm -O linux -T kernel \

-C none -a (ZRELADDR)−e(ZRELADDR)−e(ZRELADDR) \

-n 'Linux-(KERNELRELEASE)′−d(KERNELRELEASE)′−d< $@

1.3.   小结
从上面分析可知道,linux内核被bootloader拷贝到RAM后,解压代码从ZTEXTADDR开始运行(这段代码是与位置无关的PIC)。内核被解压缩到ZREALADDR处,也就是内核启动的物理地址处。相应地,内核启动的虚拟地址被设定为TEXTADDR,满足如下条件:

TEXTADDR = PAGE_OFFSET + TEXT_OFFSET

内核启动的物理地址和虚拟地址满足入下条件:

ZRELADDR == virt_to_phys(PAGE_OFFSET + TEXT_OFFSET)= virt_to_phys(TEXTADDR)

假定开发板为smdk2410,则有:

内核启动的虚拟地址

TEXTADDR     = 0xC0008000

内核启动的物理地址

ZRELADDR     = 0x30008000

如果直接从flash中启动还需要设置ZTEXTADDR地址。

2.       内核启动过程分析
内核启动过程经过大体可以分为两个阶段:内核映像的自引导;linux内核子模块的初始化。

start

Decompress_kernel()

Call_kernel

Stext:

Prepare_namespace

Do_basic_setup

init

Rest_init

Setup_arch ……

Start_kernel

_enable_mmu

Execve(“/sbin/init”))

内核启动流程图

2.1.   内核映像的自引导
这阶段的主要工作是实现压缩内核的解压和进入内核代码的入口。

Bootloader完成系统引导后,内核映像被调入内存指定的物理地址ZTEXTADDR。典型的内核映像由自引导程序和压缩的VMlinux组成。因此在启动内核之前需要先把内核解压缩。内核映像的入口的第一条代码就是自引导程序。它在arch/arm/boot/compressed/head.S文件中。

Head.S文件主要功能是实现压缩内核的解压和跳转到内核vmlinux内核的入口。Decompress_kernel(): arch/arm/boot/compressed/misc.c 和call_kernel这两个函数实现了上述功能。在调用decompress_kernel()解压内核之前,需要确保解压后的内核代码不会覆盖掉原来的内核映像。以及设定内核代码的入口地址ZREALADDR。

.text

adr   r0, LC0

ldmia       r0, {r1, r2, r3, r4, r5, r6, ip, sp}

.type       LC0, #object

LC0:              .word      LC0               @ r1

.word      __bss_start            @ r2

.word      _end                     @ r3

.word      zreladdr          @ r4

.word      _start                    @ r5

.word      _got_start              @ r6

.word      _got_end        @ ip

.word      user_stack+4096           @ sp

上面这段代码得到内核代码的入口地址,保存在r4中。

/*

* Check to see if we will overwrite ourselves.

*   r4 = final kernel address

*   r5 = start of this image

*   r2 = end of malloc space (and therefore this image)

* We basically want:

*   r4 >= r2 -> OK

*   r4 + image length <= r5 -> OK

*/

cmp r4, r2

bhs wont_overwrite

add r0, r4, #4096*1024 @ 4MB largest kernel size

cmp r0, r5

bls   wont_overwrite

mov r5, r2                    @ decompress after malloc space

mov r0, r5

mov r3, r7

bl     decompress_kernel

b     call_kernel

上面代码判断解压后的内核代码会不会覆盖原来的内核映像,然后调用内核解压缩函数decompress_kernel()。

ulg

decompress_kernel(ulg output_start, ulg free_mem_ptr_p, ulg free_mem_ptr_end_p,

int arch_id)

{

output_data            = (uch *)output_start;    /* 指定内核执行地址,保存在r4中*/

free_mem_ptr        = free_mem_ptr_p;

free_mem_ptr_end = free_mem_ptr_end_p;

__machine_arch_type    = arch_id;

arch_decomp_setup(); /*解压缩前的初始化和设置,包括串口波特率设置等*/

makecrc();           /*CRC校验*/

putstr("Uncompressing Linux...");

gunzip();            /*调用解压缩函数*/

putstr(" done, booting the kernel.\n");

return output_ptr;

}

本文转自张昺华-sky博客园博客,原文链接:http://www.cnblogs.com/sky-heaven/p/4846706.html,如需转载请自行联系原作者

ARM linux内核启动时几个关键地址【转】相关推荐

  1. linux指定内核位置,ARM linux内核启动时几个关键地址

    1.       内核启动地址 ZTEXTADDR 解压代码运行的开始地址.没有物理地址和虚拟地址之分,因为此时MMU处于关闭状态.这个地址不一定时RAM的地址,可以是支持读写寻址的flash等存储中 ...

  2. 把图形写入linux内核,如何在linux内核启动时添加显示图片

    内核版本为2.6.30.4 为了实现了在linux内核启动时不仅仅是只显示一个静态的全屏logo, 而是显示能够表示内核正在启动的进度条,因此需要能够在启动过程中直接操纵framebuffer的功能. ...

  3. 简述arm linux内核启动流程,Linux内核启动过程和Bootloader(总述)

    1.Linux内核启动过程概述 一个嵌入式 Linux 系统从软件角度看可以分为四个部分:引导加载程序(Bootloader),Linux 内核,文件系统,应用程序.其中 Bootloader是系统启 ...

  4. arm linux内核启动过程,ARM64的启动过程之(一):内核第一个脚印

    ARM64的启动过程之(一):内核第一个脚印 作者:linuxer 发布于:2015-10-10 15:06 分类:ARMv8A Arch 一.前言 kernel的整个启动过程涉及的内容很多,不可能每 ...

  5. arm linux 内核启动,Linux(ARM)内核启动地址

    最近在网上看到一位网友写得一篇文章,写得很好,加深了对Linux启动的认识,特意贴在这里: 内核编译链接过程是依靠vmlinux.lds文件,以arm为例vmlinux.lds文件位于kernel/a ...

  6. 查看Linux内核启动时调用的init函数

    Linux内核引导加载后 会调用一些初始化函数 其中有很多模块化的代码使用do_initcalls调用 这些函数通过宏pure_initcall.core_initcall.subsys_initca ...

  7. linux内核启动时添加打印日志并验证的方法

    一.相关信息 内核版本: 5.0.1 方法名:pr_info(): 添加位置: linux-5.0.1/init# vim main.c 537 asmlinkage __visible void _ ...

  8. linux卡死在选择内核界面,求助:am3352 linux内核启动时卡在 Starting kernel ...

    这是用光盘里的uImage的输出信息: U-Boot# tftp 0x82000000 bakuImage link up on port 0, speed 100, full duplex Usin ...

  9. 从linux内核启动,学习Linux内核启动过程:从start_kernel到init

    一.实验步骤: 1:运行menuos: a)cd LinuxKernel/ b)qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd root ...

最新文章

  1. AttributeError: ‘SVC‘ object has no attribute ‘_probA‘
  2. 如何用OneNote分享长微博
  3. WebService 学习之路(一):了解并使用webService
  4. C# 功能完整的单表增删改查程序
  5. c语言case后加3 5,10.程序流程控制:switch-case
  6. 4.Python算法之试探算法思想(回溯法)
  7. 正念奇迹(一则正能量)
  8. iOS CoreAnimation
  9. 关于Spring底层原理面试的那些问题,你是不是真的懂Spring?
  10. 文件夹去掉git版本控制_git 从版本控制中删除文件及.gitignore的用法
  11. 不加密,DES加密 RSA加密图
  12. Java密码学原型算法实现——第三部分:双线性对
  13. Pycharm通过ssh远程连接服务器
  14. opengGL实战——太阳系三维场景动画搭建
  15. 性能优化: 资源合并与压缩 -- 压缩(前端开发过程中 JavaScript、HTML、CSS 文件的压缩)
  16. 网管“北向接口”与“南向接口”
  17. php sapi模式,PHP中的SAPI是什么?如何实现?(图文)
  18. Ice飞冰目录结构应用入口工程配置路由配置《四》
  19. 【C++】STL简介
  20. 尚硅谷网课笔记 P361-P370

热门文章

  1. qt 4.8.4 linux,Tslib和Qt 4.8.4与在开发板上的移植
  2. 基台和种植体如何连接_成年人门牙因某些原因导致缺失该怎么办?如何修复缺失的门牙?...
  3. python ip动态代理_Scrapy 配置动态代理IP的实现
  4. linux下的tmpfs目录重启后文件全部消失
  5. unity 创建图片ui_在 Unity中 将游戏数据储存至本地文档
  6. aws 部署python lambda_python - 如何使用Aws Lambda(python)接收文件 - 堆栈内存溢出
  7. Amazon Elasticsearch Service 入门实践
  8. 【企业管理】怎么把战略和规划转化为实际行动
  9. 【项目管理】质量管理
  10. 笔记整理-信息系统工程监理(四控三管一协调)-监理规划、监理实施细则