2.移植uboot-添加2440单板,并实现NOR、NAND启动

2023-11-03

1.首先在uboot里新建单板2440

1.1将2410的单板文件夹拷贝成2440:

cd board/samsung/
cp smdk2410 smdk2440 -rf                //拷贝文件夹,

 

然后将smdk2440下的smdk2410.c改为smdk2440.c,以及修改更改好的Makefile 

1.2 将2410的头文件拷贝成2440:

cd ../../include/configs/
cp smdk2410.h smdk2440.h

 

2.新建后,还需要修改boards.cfg,使uboot支持2440单板:

仿照

smdk2410                     arm         arm920t     -                   samsung        s3c24x0

添加:

smdk2440                     arm         arm920t     -                   samsung        s3c24x0

添加后,就能够使用make smdk2440_config命令.

(该命令便会调用include/configs/smdk2440.h和board/samsung/smdk2440里的文件来配置uboot)

 

3.修改uboot系统时钟

在start.S里,uboot只设置了CLKDIVN寄存器

而2440的系统时钟需要设置两个寄存器:MPLLDIVN(设置FCLK频率)、CLKDIVN(设置分频比例),且还要设为异步模式

所以将:

      ldr   r0, =CLKDIVN
      mov r1, #3
      str   r1, [r0] 

改为:

复制代码

#define S3C2440_MPLL_400MHZ     ((0x5c<<12)|(0x01<<4)|(0x01))   //设置FCLK=400MHZ
#define MPLLCON                          0x4C000004            //设置FCLK频率

       ldr r0,=0x4C000014
       mov r1,#5                  /*FCLK:HCLK:PCLK=1:4:8  (400M:100M:50M)*/
       str r1,[r0]       

       mrc  p15, 0, r1, c1, c0                 /* 读出控制寄存器 */  
       orr   r1, r1, #0xc0000000              /* 设置为“asynchronous bus mode” */
       mcr    p15, 0, r1, c1, c0, 0             /* 写入控制寄存器 */     

       ldr r0,=MPLLCON
       ldr r1,=S3C2440_MPLL_400MHZ      
       str r1,[r0]

复制代码

 并修改board\samsung\smdk2440\Smdk2440.c里的board_early_init_f()函数,屏蔽对MPLLDIVN, LOCKTIME寄存器的设置(该函数被start.S->board_init_f()调用,这些寄存器在前面已被设置)

4.烧写到NOR上,测试

烧写测试时,由于新的uboot较大,可以使用nor上的旧uboot,通过DNW烧写新的uboot到nor上面.

步骤:

复制代码

usb 1 30000000            //使用usb下载到SDRAM上,1表示一直下载,直到完成
                         //然后打开DNW,传输新的uboot.bin给usb

protect off all           //关闭nor的写保护

erase  0   +7FFFF      //擦除nor上的 0~7FFFF地址内容, +7FFF=擦除长度=512kb,要大于新的uboot.bin才行

cp.b  30000000  0  80000         //将SDRAM上的新的uboot.bin,拷贝到nor上

复制代码

烧写完成后,重启,通过JTAG调试的读地址命令, 判断是否与新uboot文件一致

使用JTAG调试时,发现向0x30000000地址上写值出错

 

5.接下来便修改bank寄存器

将board\samsung\smdk2440\lowlevel_init.S里SMRDATA符号下:

复制代码

SMRDATA:
    .word (0+(B1_BWSCON<<4)+(B2_BWSCON<<8)+(B3_BWSCON<<12)+(B4_BWSCON<<16)+(B5_BWSCON<<20)+(B6_BWSCON<<24)+(B7_BWSCON<<28))
    .word ((B0_Tacs<<13)+(B0_Tcos<<11)+(B0_Tacc<<8)+(B0_Tcoh<<6)+(B0_Tah<<4)+(B0_Tacp<<2)+(B0_PMC))
    .word ((B1_Tacs<<13)+(B1_Tcos<<11)+(B1_Tacc<<8)+(B1_Tcoh<<6)+(B1_Tah<<4)+(B1_Tacp<<2)+(B1_PMC))
    .word ((B2_Tacs<<13)+(B2_Tcos<<11)+(B2_Tacc<<8)+(B2_Tcoh<<6)+(B2_Tah<<4)+(B2_Tacp<<2)+(B2_PMC))
    .word ((B3_Tacs<<13)+(B3_Tcos<<11)+(B3_Tacc<<8)+(B3_Tcoh<<6)+(B3_Tah<<4)+(B3_Tacp<<2)+(B3_PMC))
    .word ((B4_Tacs<<13)+(B4_Tcos<<11)+(B4_Tacc<<8)+(B4_Tcoh<<6)+(B4_Tah<<4)+(B4_Tacp<<2)+(B4_PMC))
    .word ((B5_Tacs<<13)+(B5_Tcos<<11)+(B5_Tacc<<8)+(B5_Tcoh<<6)+(B5_Tah<<4)+(B5_Tacp<<2)+(B5_PMC))
    .word ((B6_MT<<15)+(B6_Trcd<<2)+(B6_SCAN))
    .word ((B7_MT<<15)+(B7_Trcd<<2)+(B7_SCAN))
    .word ((REFEN<<23)+(TREFMD<<22)+(Trp<<20)+(Trc<<18)+(Tchr<<16)+REFCNT)
    .word 0x32
    .word 0x30
    .word 0x30

复制代码

改为:

复制代码

SMRDATA:
       .long 0x22011110    //BWSCON
       .long 0x00000700    //BANKCON0
       .long 0x00000700    //BANKCON1
       .long 0x00000700    //BANKCON2
       .long 0x00000700    //BANKCON3 
       .long 0x00000700    //BANKCON4
       .long 0x00000700    //BANKCON5
       .long 0x00018005    //BANKCON6
       .long 0x00018005    //BANKCON7
       .long 0x008C04F4    //REFRESH
       .long 0x000000B1    //BANKSIZE
       .long 0x00000030    //MRSRB6
       .long 0x00000030    //MRSRB7

复制代码

 

6.重新编译烧写uboot,发现串口已有数据,但是乱码:

 

6.1进入arch\arm\cpu\arm920t\s3c24x0\Speed.c下的get_HCLK ()函数:

 

由于我们没有配置CONFIG_S3C2440宏,所以uboot获取HCLK时钟设置波特率时,用的是CONFIG_S3C2410宏的方法

搜索CONFIG_S3C2410宏,找到位于smdk2440.h:

 

然后将smdk2440.h的CONFIG_S3C2410宏 改为: CONFIG_S3C2440宏

 

6.2编译测试

make时,发现以下几个error:

 

进入drivers/mtd/nand/s3c2410_nand.c 的72行:

 

其中nand是一个s3c2410_nand结构体:

 

该结构体如下所示:

 

从上图可以看出,只有定义了CONFIG_S3C2410宏,才能得到该结构体,而我们6.1小节里,使用的是CONFIG_S3C2440宏。

且上面的s3c2410_nand结构体和s3c2440_nand结构体的差别也很大,修改s3c2410_nand.c会很麻烦

6.3所以就直接去掉该文件,不让编译器编译即可,步骤如下所示:

1)直接进入s3c2410_nand.c的目录,打开Makefile:

 

如上图所示,需要去掉CONFIG_NAND_S3C2410宏定义才行

2)搜索CONFIG_NAND_S3C2410宏,位于include/configs/smdk2440.h:

 

如上图所示,我们直接来屏蔽CONFIG_CMD_NAND宏即可,因为该宏下的#ifdef,都是与2410相关的

3)屏蔽include/configs/smdk2440.h下的CONFIG_CMD_NAND宏定义

 

重新烧写进norflash,打印如下图所示:

 

发现无乱码了,表示nor启动成功,其中Flash: *** failed ***表示不支持norflash,因为我们只实现了重定位,并没有对nor实现写擦除等命令。

 

7.接下来便修改uboot,实现NAND启动

新的uboot链接地址位于0,且在arm-linux-ld时加了"-pie"选项, 使得u-boot.bin里多了"*(.rel*)", "*(.dynsym)",从而程序非常大,不利于从NAND启动(重定位之前的启动代码应该少于4K).

所以接下来修改代码,并取消"-pie"选项.

7.1去掉 "-pie"选项

使用grep "\-pie" * -nR找到:

arch/arm/config.mk:75:LDFLAGS_u-boot += -pie             // LDFLAGS: arm-linux-ld的参数

所以屏蔽arch/arm/config.mk文件的"LDFLAGS_u-boot += -pie"这行即可

7.2参考之前自制uboot使用的start.S, init.c来修改uboot代码

1)将以前写uboot里的init.c放入board/samsung/smdk2440目录, 并检查是否有同名函数名,若函数只在同文件使用,则添加static.并修改Makefile 

2)修改include/configs/smdk2440.h文件

将CONFIG_SYS_TEXT_BASE宏改为0x33f00000,也就是uboot重定位后的位置, 这里留了1MB空间供给uboot重定位(在反汇编中看到,代码真正总大小为700多KB(包括了bss段))

 

3)修改arch/arm/cpu/arm920t/start.S,更改重定位代码

由于nand启动时,2440未初始化之前只有前4K可读写,所以将重定位代码放在start.S的cpu_init_crit(初始化SDRAM)段后面

添加以下带红色的字段:

#ifndef CONFIG_SKIP_LOWLEVEL_INIT

       bl    cpu_init_crit

#endif

       /*重定位                          */

       ldr   sp, =(CONFIG_SYS_INIT_SP_ADDR)          //等于0x30000f80

       bic   sp, sp, #7                                                   /* 8-byte alignment for ABI compliance */

    

       mov r0,#0                         //r0->src

       ldr r1,_TEXT_BASE          //_TEXT_BASE : 0x33f00000

       ldr r2,_bss_start_ofs       // _bss_start_ofs:  __bss_start - _start   (有效代码大小)

       bl copy_code_to_sdram     //该函数首先会初始化nand控制器,然后复制代码到SDRAM连接地址dest上

       bl clear_bss                         //清除bss段(参考自制uboot章节)

 

       ldr pc,=call_board_init_f                 //绝对跳转,跳到SDRAM上执行

call_board_init_f:

       ldr   r0,=0x00000000

       bl    board_init_f

上面的_TEXT_BASE,在start.S靠前处定义:

由于它位于靠前处,保证了_TEXT_BASE存在前4k空间里,若直接使用ldr r1,=CONFIG_SYS_TEXT_BASE,编译器可能会将这个宏定义放在SDRAM上,则会出错

4)重定位写在前面了,所以我们还要删除start.S后面的relocate_code重定位段,清除BSS段

board_init_r位置处的代码,改为如下所示:

复制代码

/* void relocate_code (addr_sp, gd, addr_moni)*/
.globl      relocate_code
relocate_code:

       mov r4, r0      /* save addr_sp */       
       mov    sp, r4
       mov r0, r1      /* save addr of gd */
       mov r1, r2      /* save addr of destination */
       bl   board_init_r        //进入uboot第二阶段代码

复制代码

 

7.3修改board_init_f()函数(位于arch/arm/lib/board.c)

本节添加的uboot重定位是直接以基地址0x33F00000开始的, 在上一章分析出,board_init_f()函数划分uboot重定位所在区域时,是通过动态划分的.

所以修改board_init_f()函数的第113行:

复制代码

       /*
        * reserve memory for U-Boot code, data & bss
        * round down to next 4 kB limit
        */

       //addr -= gd->mon_len;                     //屏蔽该行
       //addr &= ~(4096 - 1);                     //屏蔽该行

     addr=CONFIG_SYS_TEXT_BASE;           //0x33f00000,添加该行     

复制代码

 

7.4修改链接脚本

把start.S, init.c(实现重定位), lowlevel.S(实现初始化SDRAM)等文件放在最前面

vi arch/arm/cpu/u-boot.lds

添加以下带红色的字段:

     . = ALIGN(4);

        .text :

        {

                __image_copy_start = .;

                CPUDIR/start.o (.text)              //CPUDIR为arch/arm/cpu/arm920t目录

                board/samsung/smdk2440/libsmdk2440.o (.text)  

                *(.text)

        }

libsmdk2440.o是将smdk2440单板目录下的所有*.c,*S文件编译后,连接成一个库文件.

8.然后通过旧的uboot,将新的uboot烧写到nand

usb 1 30000000                             //先下载到SDRAM上

nand erase 0  0x80000                      //擦除512kb,必须大于新的uboot

nand write 30000000   0  0x80000           //将SDRAM上的新uboot写入nand中

烧写后,如下图所示:

 

nand启动便实现完成了,上面的Flash: *** failed *** 是属于uboot第二阶段函数board_init_r()里的代码,表示不支持nor flash,不能实现读,写,擦除等命令

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

2.移植uboot-添加2440单板,并实现NOR、NAND启动 的相关文章

  • 1.3 C++ 关键字升级

    实用性增强 遍历数组的方式 关键字的升级 register vs volatile 仅能修饰局部变量 不能修饰全局变量和函数 修饰的变量不能通过 取地址 C 当对register变量取地址时 会将该变量重新保存到内存中 寄存器变量 提高运行
  • HashMap和HashSet

    一 什么是HashSet HashSet实现了Set接口 它不允许集合中有重复的元素 而且集合中的元素都是无序的 在将对象储存在HashSet之前 要确保对象重写了equals 方法和hashCode 方法 这样才能比较对象是否相等 以确保
  • 服务器级的kvm维修,企业级KVM解决方案 Altusen KM0932评测

    IT168评测中心 KVM多电脑切换器是一种可以通过由单一键盘 Keyboard 显示器 Video 及鼠标 Mouse 组成的控制端集中管理多台电脑的硬件装置 是现代数据中心的基础设备之一 使用KVM可以有效地节约机房空间 并有效地提升机
  • Linux 查看CPU架构及内核版本

    涉及arch命令和 proc version文件 1 查看CPU架构 有些软件的安装需要和CPU架构相匹配 如JDK等等 所以需要确定主机的CPU架构类型 可使用命令arch查看Linux系统的CPU架构 如下 arch 2 查看内核版本

随机推荐

  • 国家网络安全宣传周知识竞赛活动小程序界面分享

    国家网络安全宣传周知识竞赛活动小程序界面分享
  • el-menu多级动态菜单渲染

    使用el menu时往往会遇到动态渲染使用场景 安装和引用element暂不赘叙 对其使用方法不是很熟悉的小伙伴可以去官网按照指引完成使用 先简单叙述一下渲染思路 首先我们要知道要渲染一个多级菜单的过程实际上就是对后端给我们的多层数据格式进
  • 后端研发Redis必知必会

    本篇内容完全偏向于实践 也是后端开发常用到的知识 关于Redis原理与概念性的内容会另起一篇来说明 可以根据目录来选择自己所需要的内容来阅读 文章目录 1 redis安装 1 1 ubuntu上安装redis 1 2 centos上安装re
  • 少儿编程课程和乐高机器人有什么不同

    少儿编程课程和乐高机器人有什么不同 现在很多的家长开始给孩子选择课外学习的课程 可以说是越来越高科技 就拿现在人工智能的发展来说 很多的家长开始给孩子选择少儿编程的课程来让孩子学习 但是很多的家长想要孩子去学习的时候 不知道少儿编程和乐高机
  • Unity开发详解之旋转、移动、碰撞(3/6)

    在前两篇中 我们已经创建好了场景和玩家对象 下面让玩家对象动起来 玩家对象旋转 using System Collections using System Collections Generic using UnityEngine publ
  • QString::arg: Argument missing的报错解决办法

    当Qt sqlite 插入语句query exec insert into qper values 1 2 3 4 5 6 arg 参数1 arg 参数2 arg 参数3 arg 参数4 arg 参数5 arg 参数6 出现报错时 可将参数
  • inputstreamreader用法

    InputStreamReader 是 Java 中的一个类 它可以将字节输入流转换为字符输入流 它可以读取字节输入流 并使用指定的字符集将字节解码为字符 它的常用用法如下 从文件读取内容 FileInputStream fis new F
  • 807-C++多继承下,派生类对象有几张虚函数表?

    C 多继承下 派生类对象有几张虚函数表 我们看下面这个示例 include
  • PostgreSQL(数据库)资料和分布式系统(Distributed System)资料与Go语言资料整理

    PostgreSQL Documentation 介绍 PostgreSQL官方文档主页 PG技术沙龙ppt 介绍 2013年4月PG技术沙龙PPT PG9 3中文翻译文档 介绍 PG9 3中文翻译文档 谢谢 skykiker提供 Post
  • Android Studio ADB 环境变量配置

    Android 引用 SQLite关系库 需要用到ADB工具 如下为ADB环境变量配置 1 找到文件 sdk 下的 tools 和 platform tools 并复制其路径 2 打开环境变量配置 3 新建系统变量 命名为 Android
  • jQuery鼠标右击事件显示弹框

    content on contextmenu function e e preventDefault return false content on mousedown list function e console log e which
  • 【ENVI遥感影像分类】 监督、非监督分类

    基于光谱的分类方法 1 概要 灰度分割 非监督分类 ISODATA K Means 监督分类 基于传统统计分析分类器 平行六面体 最小距离 马氏距离 最大似然 分类器 基于人工智能分类器 神经网络 分类器 基于模式识别分类器 支持向量机 模
  • linux驱动37:kmalloc内存分配

    kmalloc和malloc类似 可被阻塞 分配的内存保持原有数据 分配的区域在物理内存中也是连续的 头文件
  • 卡尔曼滤波-卡尔曼滤波全篇讲解

    卡尔曼滤波 个人理解 两种信号 状态变量 按照人类经验预测的状态 观测变量 根据生活工具测量的实际状态 噪声 噪声分布 噪声表达 滤波器的计算原型 先验估计 后验估计与测量状态变量 计算增益系数K 数据融合基础变量 先验估计 测量状态变量
  • Apache Beam构建流处理任务

    最近做的一个项目需要用到Google云平台的Dataflow来进行数据处理 因此研究了一下相关的文档 了解到Dataflow是基于Apache beam来进行流程的编排 Beam支持多种不同的Runner 除了Dataflow Beam还支
  • 基于Docker安装及运行Mongodb

    1 拉取镜像 docker pull mongo latest 2 运行mongodb docker run itd name mongo p 27017 27017 mongo auth 3 进入mongodb容器 docker exec
  • 《只管去做》-如何做靠谱的年度计划

    五年后的一封信 调动五感去写 要找到怦然心动的感觉 世界上的人4种状态 空想家 这些人总是有很多想法 但真正落实去做的很少 回应者 这样的人对未来没有任何想法 对现状也没有掌控力 是一种混吃等死的状态 每一天都过得差不多 也安于现状 执行者
  • leecode刷题:1.两数之和

    给定一个整数数组 nums 和一个整数目标值 target 请你在该数组中找出 和为目标值 target 的那 两个 整数 并返回它们的数组下标 你可以假设每种输入只会对应一个答案 但是 数组中同一个元素在答案里不能重复出现 你可以按任意顺
  • Java基础:序列化与反序列化

    Java序列化是指把Java对象转换为字节序列的过程 而Java反序列化是指把字节序列恢复为Java对象的过程 序列化的使用场景 永久性保存对象 保存对象的字节序列到本地文件或者数据库中 通过序列化以字节流的形式使对象在网络中进行传递和接收
  • 2.移植uboot-添加2440单板,并实现NOR、NAND启动

    1 首先在uboot里新建单板2440 1 1将2410的单板文件夹拷贝成2440 cd board samsung cp smdk2410 smdk2440 rf 拷贝文件夹 然后将smdk2440下的smdk2410 c改为smdk24