s3c2440移植Linux内核,移植Linux-3.4.2内核到S3C2440

发布者:清新微笑最新更新时间:2022-05-19 来源: eefocus关键字:s3c2440  移植  Linux内核 手机看文章 扫描二维码
随时随地手机看文章

一、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..

*/

init_post();

return 0;

}

kernel_init先调用了prepare_namespace();然后调用了init_post函数

在prepare_namespace()函数里 调用mount_root()函数,挂载根文件系统;

三、移植linux3.4.2到JZ2440

1、解压tar xjf linux-3.4.2.tar.bz2

2、进入解压后的文件目录,修改顶层Makefile

vim Makefile

修改架构为 ARM 以及编译器

ARCH=arm

CROSS_COMPILE=arm-linux-

3、选择默认配置

find -name"*defconfig"

4、在解压后文件目录下,配置,生成.config文件

make s3c2410_defconfig

5、查看支持的单板

vim .config

6、编译

make uImage

7、u-boot2012里默认的是193机器ID,设置机器ID为362使用SMDK2440,在uboot中设置机器ID

set machid 16a

save

8、在uboot中设置启动行参数并修改smdk2440单板的晶振信息12M

bootargs noinitrd root=/dev/nfs nfsroot=192.168.1.112:/opt/filesystem ip=192.168.1.130:192.168.1.112:192.168.1.1:255,255,255,0::eth0:off init=/linuxrc console=ttySAC0,115200

修改文件mach-smdk2440.c的晶振信息12M

static void __init smdk2440_map_io(void)

{

s3c24xx_init_io(smdk2440_iodesc, ARRAY_SIZE(smdk2440_iodesc));

s3c24xx_init_clocks(12000000);

s3c24xx_init_uarts(smdk2440_uartcfgs, ARRAY_SIZE(smdk2440_uartcfgs));

}

四、修改分区

我们经常用的内核打印分区信息如下

Creating 4 MTD partitions on "NAND":

0x000000000000-0x000000040000 : "bootloader"

0x000000040000-0x000000060000 : "params"

0x000000060000-0x000000460000 : "kernel"

0x000000460000-0x000010000000 : "rootfs"

这些分区是通过在文件linux-2.6.22.6archarmplat-s3c24xx/Common-smdk.c设置的

/* NAND parititon from 2.4.18-swl5 */

static struct mtd_partition smdk_default_nand_part[] = {

[0] = {

.name    = "bootloader",

.size    = SZ_256K,

.offset    = 0,

},

[1] = {

.name    = "params",

.offset = MTDPART_OFS_APPEND,

.size    = SZ_128K,

},

[2] = {

.name    = "kernel",

.offset = MTDPART_OFS_APPEND,

.size    = SZ_4M,

},

[3] = {

.name    = "rootfs",

.offset    = MTDPART_OFS_APPEND,

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

上一篇:ARM 学习笔记 (三) S3C2440 MMU 配置
下一篇:U-boot 在 mini2440-S3C2440 上的移植(4):U-boot第二阶段代码:修改初始化代码

推荐阅读最新更新时间:2024-11-09 10:15

基于OHCI协议的S3C2440 USB Host Driver之协议介绍(二)
一 HCD的职责: 1.HCD负责HC的操作。 HCD可以直接与HC的可操作寄存器通信,且在HCCA中建立中断ED表头指针。HCD维护HC的状态,表处理指针,表处理使能,中断使能。 2.带宽分配。 HCD负责对USB访问进行调度。HCD给每一个周期端点分配一定可用的带宽。如果没有足够的带宽可用,一个新连接的周期端点不允许访问总线。 一部分带宽分给非周期传输。这会确保在每帧里一定量的批量和控制传输会发生。每帧是1ms。 每帧的开始,HC发起SOF给USB总线。HC紧接着服务非周期列表。然后服务周期列表。还有剩余时间的话,再次服务非周期列表。 3列表管理。 USB传输机制以通过连接ED表上的TD。HCD
[单片机]
基于OHCI协议的<font color='red'>S3C2440</font> USB Host Driver之协议介绍(二)
mini2440 移植 Java虚拟机的错误处理
大都数朋友都会以下面这篇文摘《移植Java虚拟机到mini2440》http://www.linuxidc.com/Linux/2011-08/41912.htm 作为参考,不过里面有几个错误: 首先,在 “ 修改 segvhandler_arch.c 文件 gedit ../../src/linux-arm/javavm/runtime/segvhandler_arch.c 修改 #define ucontext asm_ucontext #include 为 //#define ucontext asm_ucontext //注释掉这句 #include //去掉arm/ ”
[单片机]
S3C2440——键盘中断服务程序
S3C2440键盘模块电路 一共有S1、S2、S3、S4个按键,分别对应EINT19、EINT2、EINT0、EINT11这四个中断源。中断框架及服务程序如下: 中断响应,取出触发中断的键盘编号(1-4)放到R5中 ;文件ASM_Interrupt.s ;(1)设置中断向量表 Mode_USR EQU 0x50 ;IRQ中断开放,FIQ中断关闭 Mode_FIQ EQU 0xD1 ;关闭IRQ、FIQ中断 Mode_IRQ EQU 0xD2 ;关闭IRQ、FIQ中断 Mode_SVC EQU 0xD3 ;关闭IRQ、FIQ中断 GET 2440Reg_addr.
[单片机]
<font color='red'>S3C2440</font>——键盘中断服务程序
文件系统的移植(一):Initramfs文件系统的移植
内 核 版 本:linux-3.0 u-boot版本:u-boot-2010.09 开发板型号:FL2440 1、对根文件系统进行修改 (一)的前提下不需要修改 2、添加内核对 initramfs 的支持 $ vt100 $ make menuconfig General setup --- 注:其中:../../rootfs_tree为自己的文件系统相对于内核的路径。 修改Makefile $ vim Makefile 修改195、196行 注:196行为自己的交叉编译器的路径 $ make 编译过程省略 $ ls ky-initramfs.bin为编译好的内核。
[单片机]
文件系统的<font color='red'>移植</font>(一):Initramfs文件系统的<font color='red'>移植</font>
S3C2440 spi驱动简单测试
这两天参考网上的资料,自己写了个SPI的驱动,并实际测试通过。 硬件平台:mini2440 用的是S3C2440 的SPI1(共有2个SPI模块) 操作系统:linux-2.6.32.2 测试方法:将SPI的MISO与MOSI管脚短路,这样读数据的时候第一个发出的dummy字节即为收到的字节。 下面是驱动的源代码(mini2440_spi.c): /***************************************************/ #include linux/irq.h #include linux/miscdevice.h #include linux/delay.h #incl
[单片机]
S3C2440内核蜂鸣器驱动解读
首先介绍需要的一些头文件的位置 linux-2.6.32.内核重要文件目录: linux-2.6.32.2/arch/arm/mach-s3c2410/include/mach/regs-gpio.h linux-2.6.32.2/arch/arm/plat-s3c24xx/gpio.c linux-2.6.32.2/linux/asm-generic/io.h linux-2.6.32.2/include/linux/wait.h asm -- linux-2.6.32.2/linux/asm-generic mach -- linux-2.6.32.2/arch/arm/mach-s3c2410/include/mach
[单片机]
<font color='red'>S3C2440</font>内核蜂鸣器驱动解读
s3c2440裸机-异常中断(四. irq之外部中断)
中断前: 中断产生后: 问题案例: 我们想实现一个按键点灯程序,我们知道有以下两种方案: 1.轮询方案:轮询检测按键的电平状态,当检测到被按下后,对应的gpio会拉低,点亮对应的led;(略) 2.中断方案:将按键配置成外部中断源,当有按键按下,触发中断,在中断服务程序(isr)中去完成点灯。 下面开始写代码: 一.中断初始化 1)中断源设置 我们用按键作为外部中断源,我们把按键对应的gpio配置成中断引脚,当按键按下,相应的gpio产生了电平跳变,就会触发外部中断。 我们想达到按下按键灯亮,松开按键灯灭这种效果(配成双边沿触发,按下的时候产生下降沿中断,进行点亮,松开产生上升沿中断,进行熄灭)。当然也可做成按一下
[单片机]
<font color='red'>s3c2440</font>裸机-异常中断(四. irq之外部中断)
ok6410上移植madplay
交叉编译时与上篇过程相同,但是在配置的时候要做些修改。 1.安装zlib 用交叉编译工具编译zlib,并且把库生成到交叉编译环境的库目录下 ./configure --prefix=/usr/local/arm/4.4.1/arm-none-linux-gnueabi --prefix是便于寻找头文件和库文件 修改Makefile. CC=arm-linux-gcc AR=arm-linux-ar rc RANLIB=arm-linux-ranlib make make install 2.编译libid3tag ./configure CC=arm-linux-gcc --host=arm-linux --disable-
[单片机]
小广播
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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