移植Linux-3.4.2内核到S3C2440

发布者:HeavenlyWonder最新更新时间:2024-06-17 关键字:移植  2内核  S3C2440 手机看文章 扫描二维码
随时随地手机看文章

一、BootLoader引导内核过程

    1、Bootloader的工作


    1.1、将内核读入内存

    1.2、保存内核启动参数到指定位置,内核启动时去这个位置解析参数

    1.3、启动内核、传入机器ID

二、内核的启动流程

内核首要目的是挂载根文件系统,启动应用程序,内核启动的过程大致为以下几步:

1.检查CPU和机器类型

2.进行堆栈、MMU等其他程序运行关键的东西进行初始化

3.打印内核信息

4.执行各种模块的初始化

5.挂接根文件系统

6.启动第一个init进程

对于ARM的处理器,内核第一个启动的文件是arc/arm/kernel下面的head.S文件

第一阶段:

首先截取部分head.S文件

ENTRY(stext)

 THUMB(    adr    r9, BSYM(1f)    )    @ Kernel is always entered in ARM.
 THUMB(    bx    r9        )    @ If this is a Thumb-2 kernel,
 THUMB(    .thumb            )    @ switch to Thumb now.
 THUMB(1:            )

    setmode    PSR_F_BIT | PSR_I_BIT | SVC_MODE, r9 @ ensure svc mode
                        @ and irqs disabled
    mrc    p15, 0, r9, c0, c0        @ get processor id
    bl    __lookup_processor_type        @ r5=procinfo r9=cpuid
    movs    r10, r5                @ invalid processor (r5=0)?
 THUMB( it    eq )        @ force fixup-able long branch encoding
    beq    __error_p            @ yes, error 'p'

#ifdef CONFIG_ARM_LPAE
    mrc    p15, 0, r3, c0, c1, 4        @ read ID_MMFR0
    and    r3, r3, #0xf            @ extract VMSA support
    cmp    r3, #5                @ long-descriptor translation table format?
 THUMB( it    lo )                @ force fixup-able long branch encoding
    blo    __error_p            @ only classic page table format
#endif

第一步,执行的是__lookup_processor_type,这个函数是检查处理器型号,它读取你的板子的CPU型号与内核支持的处理器进行比较看是否能够处理。

第二步,检查机器型号,它会读取你bootloader传进来的机器ID和他能够处 理的机器ID进行比较看是否能够处理。内核的ID号定义在arc/arm/tool/mach_types文件中MACH_TYPE_xxxx宏定义。内 核究竟就如何检查是否是它支持的机器的呢?实际上每个机器都会在/arc/arm/mach-xxxx/smdk-xxxx.c文件中有个描述特定机器的 数据结构,


MACHINE_START(S3C2440, 'SMDK2440')
    /* Maintainer: Ben Dooks */
    .atag_offset    = 0x100,

    .init_irq    = s3c24xx_init_irq,
    .map_io        = smdk2440_map_io,
    .init_machine    = smdk2440_machine_init,
    .timer        = &s3c24xx_timer,
    .restart    = s3c244x_restart,
MACHINE_END

MACHINE_START和 MACHINE_END实际上被展开成一个结构体

#defineMACHINE_START(_type,_name)                 
staticconst struct machine_desc __mach_desc_##_type       
 __used                                             
 __attribute__((__section__('.arch.info.init')))= {     
      .nr          =MACH_TYPE_##_type,           
      .name            =_name, 
       
#defineMACHINE_END                           
}; 

于是上面的数据结构就被展开为

staticconst struct machine_desc __mach_desc_S3C2440     
 __used                                             
 __attribute__((__section__('.arch.info.init')))= {     
      .nr          =MACH_TYPE_S3C2440,           
      .name            =”SMDK2440”,}; 
.phys_io  = S3C2410_PA_UART
      .io_pg_offst    = (((u32)S3C24XX_VA_UART) >> 18) & 0xfffc, 
      .boot_params  = S3C2410_SDRAM_PA + 0x100, 
 
      .init_irq  =s3c24xx_init_irq, 
      .map_io          =smdk2440_map_io, 
      .init_machine  = smdk2440_machine_init, 
      .timer            =&s3c24xx_timer, 
 

每个机器都会有一个machine_desc__mach_desc结构,内核通过检查每个machine_desc__mach_desc的nr 号和bootloader传上来的ID进行比较,如果相同,内核就认为支持该机器,而且内核在后面的工作中会调用该机器的 machine_desc__mach_desc_结构中的方法进行一些初始化工作。

第三步,创建一级页表

第四步,在R13中保存__switch_data 这个函数的地址,在第四步使能mmu完成后会跳到该函数执行。

第五步,执行的是__enable_mmu,它是使能MMU,这个函数调用了__turn_mmu_on函数,让后在_turn_mmu_on在最 后将第三步赋给R13的值传给了PC指针 (mov    pc, r13),于是内核开始跳到__switch_data这个函数开始执行。

我们再来看arch/arm/kenel/head-common.S这个文件中的__switch_data函数

/*
 * The following fragment of code is executed with the MMU on in MMU mode,
 * and uses absolute addresses; this is not position independent.
 *
 *  r0  = cp#15 control register
 *  r1  = machine ID
 *  r2  = atags/dtb pointer
 *  r9  = processor ID
 */
    __INIT
__mmap_switched:
    adr    r3, __mmap_switched_data

    ldmia    r3!, {r4, r5, r6, r7}
    cmp    r4, r5                @ Copy data segment if needed
1:    cmpne    r5, r6
    ldrne    fp, [r4], #4
    strne    fp, [r5], #4
    bne    1b

    mov    fp, #0                @ Clear BSS (and zero fp)
1:    cmp    r6, r7
    strcc    fp, [r6],#4
    bcc    1b

 ARM(    ldmia    r3, {r4, r5, r6, r7, sp})
 THUMB(    ldmia    r3, {r4, r5, r6, r7}    )
 THUMB(    ldr    sp, [r3, #16]        )
    str    r9, [r4]            @ Save processor ID
    str    r1, [r5]            @ Save machine type
    str    r2, [r6]            @ Save atags pointer
    bic    r4, r0, #CR_A            @ Clear 'A' bit
    stmia    r7, {r0, r4}            @ Save control register values
    b    start_kernel
ENDPROC(__mmap_switched)

    .align    2
    .type    __mmap_switched_data, %object
__mmap_switched_data:
    .long    __data_loc            @ r4
    .long    _sdata                @ r5
    .long    __bss_start            @ r6
    .long    _end                @ r7
    .long    processor_id            @ r4
    .long    __machine_arch_type        @ r5
    .long    __atags_pointer            @ r6
    .long    cr_alignment            @ r7
    .long    init_thread_union + THREAD_START_SP @ sp
    .size    __mmap_switched_data, . - __mmap_switched_data

这个函数做的工作是,复制数据段清楚BBS段,设置堆在指针,然后保存处理器内核和机器内核等工作,最后跳到start_kernel函数。于是内核开始执行第二阶段。

第二阶段:

init/目录下的main.c的start_kernel函数


asmlinkage void __init start_kernel(void)

在start_kernel首先是打印内核信息,然后对bootloader传进来的一些参数进行处理,再接着执行各种各样的初始化,在这其中会初始化控制台。最后会调用rest_init();

我们再来看rest_init()函数


static noinline void __init_refok rest_init(void)

他启动了kernel_init这个函数,再来看kerne_init函数

static int __init kernel_init(void * unused)
{
    /*
    * Wait until kthreadd is all set-up.
    */
    wait_for_completion(&kthreadd_done);

    /* Now the scheduler is fully set up and can do blocking allocations */
    gfp_allowed_mask = __GFP_BITS_MASK;

    /*
    * init can allocate pages on any node
    */
    set_mems_allowed(node_states[N_HIGH_MEMORY]);
    /*
    * init can run on any cpu.
    */
    set_cpus_allowed_ptr(current, cpu_all_mask);

    cad_pid = task_pid(current);

    smp_prepare_cpus(setup_max_cpus);

    do_pre_smp_initcalls();
    lockup_detector_init();

    smp_init();
    sched_init_smp();

    do_basic_setup();

    /* Open the /dev/console on the rootfs, this should never fail */
    if (sys_open((const char __user *) '/dev/console', O_RDWR, 0) < 0)
        printk(KERN_WARNING 'Warning: unable to open an initial console.n');

    (void) sys_dup(0);
    (void) sys_dup(0);
    /*
    * check if there is an early userspace init.  If yes, let it do all
    * the work
    */

    if (!ramdisk_execute_command)
        ramdisk_execute_command = '/init';

    if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) {
        ramdisk_execute_command = NULL;
        prepare_namespace();
    }

    /*
    * Ok, we have completed the initial bootup, and
    * we're essentially up and running. Get rid of the
    * initmem segments and start the user-mode stuff..
    */

[1] [2]
关键字:移植  2内核  S3C2440 引用地址:移植Linux-3.4.2内核到S3C2440

上一篇:移植u-boot-2010.12.tar.bz2版本到s3c2440过程
下一篇:移植u-boot-2012.04.01到JZ2440

推荐阅读最新更新时间:2024-11-13 19:19

arm-linux-gcc4.4.3编译s3c2410平台linux内核
一、首先下载linux内核: linux-2.6.14.tar.bz2 下载地址: http://www.kernel.org/pub/linux/kernel/v2.6/linux-2.6.14.tar.bz2 二、解压linux-2.6.14.tar.bz2: tar -jxvf linux-2.6.14.tar.bz2 三、配置Makefile文件: 1、打开源码根目录下的Makefile文件: gedit Makefile 修改Makefile文件: ARCH ?= arm CROSS_COMPILE ?= arm-linux- 2、查看目录结构: ls -l linux-2.6.14/arch/arm/ d
[单片机]
搭载骁龙W5可穿戴平台,OPPO Watch 4 Pro持续引领全智能可穿戴旗舰
8月29日,OPPO Watch系列新品OPPO Watch 4 Pro正式发布。全新OPPO Watch 4 Pro搭载骁龙W5可穿戴平台,凭借全面领先的软硬件实力表现,打造极致使用体验,并持续引领全智能可穿戴旗舰。 OPPO Watch 4 Pro出众的智能体验背后是强大的底层平台支持。骁龙W5可穿戴平台采用业界领先的4纳米制程工艺,集成四核Cortex-A53 CPU,频率达到1.7GHz,配合Adreno 702 GPU以及升级的内存、摄像头和音频/视频模块,与前代可穿戴平台相比,性能提升2倍,特性增加2倍,实现丰富的沉浸式交互用例和极致流畅的操作体验。OPPO Watch 4 Pro在强大硬件性能加持下,支持多
[物联网]
搭载骁龙W5可穿戴平台,OPPO Watch <font color='red'>4</font> Pro持续引领全智能可穿戴旗舰
S3C2440触摸屏控制总结
触摸屏控制原理,其实与ADC读取一个滑动变阻器中间触点电压的原理一样。只不过,读取触摸屏的X、Y方向上的电压需要两次,而且需要设置其工作模式以实现一个ADC读取两个通道的电压。 S3C2440的ADC控制是很简单的,与普通单片机控制ADC的方法没有多大区别。大概的操作步骤都是如下所示。 (1)设置控制寄存器ADCCON (2)启动ADC转换 (3)等待转换结束 (4)读取转换结果 但是,S3C2440的触摸屏控制有什么工作模式,如等待中断模式,分离的x/y轴坐标转换模式,自动(连续)x/y轴坐标转换模式,普通转换模式。通过官方芯片资料理解这些模式,似乎还是一头雾水。经过测试,我对这些模式有了初步的认识。 (一) ADC&TC模式分
[单片机]
STM32基础知识2-分享PWM输入模式捕捉4路PWM波形的周期和占空比
前几天分享过一个帖子,因为网速原因没有上传源码,看到有人回复我说实验不成功,那么好吧!这是我的错误,这次就将源码和我的心得体会分享出来,供大家下载测试和成长。我其实也是菜鸟一个,如果讲解的地方有啥不对,或是程序设计的不好,欢迎大家提出意见,让我们一起来学习进步。 PWM输入是输入捕获的一个特殊应用,输入捕获就是当连接到定时器的引脚上产生电平变化时对应的捕获装置会立即将当前计数值复制到另一个寄存器中。你可以开启捕获中断然后在中断处理函数中读出保存的计数值。 与输入捕获不同的是PWM输入模式会将同一个输入信号(TI1或TI2)连接到两个捕获装置(IC1和IC2)。这两个捕获装置一个捕获上升沿一个捕获下降沿。TI1FP1、TI2
[单片机]
STM32基础知识<font color='red'>2</font>-分享PWM输入模式捕捉<font color='red'>4</font>路PWM波形的周期和占空比
飞思卡尔小基站解决方案满足4G时代移动互联需求
小基站势如破竹 最近,中国三大移动运营商铺天盖地的4G广告让人感觉到4G时代真的来临,而且来的比3G更猛烈些。随着4G的普及,4G网络中传播的数据量是不可估量的,基站作为数据传输的中转站,其规模、形式、性能也需要进一步革新。大型基站固然重要,如目前最普遍的宏基站以及大为热门的云RAN,而现在的手机基站网络越来越复杂,基站形态越来越多,3G和4G的发展催生了大量对小型基站的需求,它们在帮助移动运营商应对日益复杂的网络容量挑战方面发挥着关键作用。小型化的基站主要有城域基站、微微蜂窝基站(Picocell)以及更小一点的毫微微蜂窝基站(Femtocell,又称飞蜂窝基站或家庭基站,相当于在一个公司里面用Wi-Fi)。 小基站
[网络通信]
飞思卡尔小基站解决方案满足<font color='red'>4</font>G时代移动互联需求
卢伟冰:价格一样你选择8GB LPDDR5还是12GB LPDDR4X
IT之家3月15日消息 今年2月份发布的小米10手机全系搭载了骁龙865处理器+LPDDR5,小米曾号称是骁龙865+ LPDDR5内存+UFS 3.0存储是旗舰机绝配。3月份,小米的独立品牌Redmi将发布Redmi K30 Pro了。看起来Redmi也开始预热Redmi K30 Pro了。   3月14日晚间,小米中国区总裁、Redmi品牌总经理卢伟冰在微博发问,假设价格一样,你会选择8GB LPDDR5还是选择 12GB LPDDR4X?这里当然是指内存了。   根据爆料消息,Redmi K30 Pro可能还会有变焦版,内存配置可能会有变化。你认为8GB LPDDR5、12GB LPDDR4X哪个好一些?  
[手机便携]
卢伟冰:价格一样你选择8GB LPDDR5还是12GB LPDDR<font color='red'>4</font>X
夏普业绩拼V字型恢复!将4年来首度转盈、纯益超预期
夏普(Sharp)26日于日股盘后公布中期(2017-2019年度)营运计划,表示业绩将实现V字型回复(2016年度业绩为V字底部),其中因中小尺寸面板、液晶电视销售增加,加上将持续实施结构改革措施,故2017年度(2017年4月-2018年3月)合并营收预估将年增22.4%至2.51兆日圆;显示本业获利状况的合并营益将大增44.1%至900亿日圆;显示最终获利状况的合并纯益预估为590亿日圆(2016年度为净损248亿日圆),将为4年来首度转亏为盈、且金额远优于金融情报服务公司QUICK事前所作调查382亿日圆的预估值。 夏普并指出,将创造8K面板需求、早期推出OLED/FFD等新面板、并将扩大海外TV事业、强化8K生态系统
[手机便携]
移植u-boot-2010.03问题 --- 网卡DM9000
1,找到u-boot-2010.03/include/configs/smdk6410.h //#define CONFIG_NET_MULTI //#define CONFIG_CS8900 /*we have aCS8900on-board*/ //#define CONFIG_CS8900_BASE 0x18800300 //#define CONFIG_CS8900_BUS16 /*follow the Linux driver*/ /**然后添加DM9000网卡的宏定义---ip以及子网掩码等根据自己的需要修改***/ #defineCONFIG_NET_MULTI 1 #defineCONFIG_DM900
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved