正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-8.2-链接脚本

本文主要是介绍正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-8.2-链接脚本,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 前言:

本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM(MX6U)裸机篇”视频的学习笔记,在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。

引用:

正点原子IMX6U仓库 (GuangzhouXingyi) - Gitee.com

《【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.5.2.pdf》第 8.1 章

《正点原子资料_A盘/02开发板原理图/IMX6ULL_MINI_V2.2(Mini底板原理图).pdf》

正点原子资料下载中心 — 正点原子资料下载中心 1.0.0 文档

 资料盘 开发板资料链接: https://pan.baidu.com/s/1j5Jzbdx9i-g0cWIi3wf2XA 提取码:ag1u

正文:

本文是 “正点原子[第二期]Linux之ARM(MX6U)裸机篇--第8.2讲” 的读书笔记。第8.2 介绍了编译器连接脚本 .lds 文件的语法。并给出了一个例子,如何使用链接脚本来实现我们上一节C语言 LED 驱动程序实验里指定 start.o main.o 的文件链接顺序。

0. 链接脚本简介

在上一节C语言LED灯驱动实验的 Makefile 里,我们链接代码时使用了如下语句

arm-linux-gnueabihf-ld -Ttext 0x87800000 -o led.elf $^

上面语句是通过 "-Ttext" 来指定连接地址是 0x87800000 的,这样的话所有文件都会链接到以0x87800000  为起始地址的区域。但是有时候我们很多文件需要连接到指定的区域,或者叫做段里面,比如在 Linux 里面初始化函数都会放到 .init 段里面。因此我们需要能够自定义一些段,这些段的起始地址我们可以自由指定,同样的我们也可以指定一个文件或者函数应该放在哪个段里面去。要完成这个功能我们就需要使用到链接脚本,看名字就知道链接脚本主要用于链接的,用于描述文件应该如何被链接在一起形成最终可执行文件。其主要目的是描述输入文件中的段如何被映射到输出文件中,并且控制输出文件中的内存排布。比如,我们编译生成的文件一般都包含 .txt 段,.data 段等等。

1. 链接脚本的语法

链接脚本的语法很简单,就是编写一系列的命令,这行命令组成了链接脚本,每个命令是一个带有参数的关键字或者一个对符号的赋值,可以使用分号分隔命令。像文件名之类的字符串可以直接键入,也可以使用通配符“*”。最简单的链接脚本可以值包含一个命令 "SECTIONS",我们可以在这一个 "SECTIONS" 里面描述输出文件的内存布局。我们一般编译出来的代码都包含 .text, .data, .bss, .rodata 这四个段,假设现在代码要被链接到 0x10000000 这个地址,数据要被链接到 0x30000000 这个地方,下面就是完成此功能的最简单的链接脚本:

SECTIONS {. = 0x10000000;.text : { *(.text) }. = 0x30000000;.data ALIGN(4) : { *(.data) }.bss ALIGN(4) : { *(.bss) }
}

第一行写了一个关键字 "SECTIONS" ,后面跟了一个大括号,这个大括号和第七行的大括号是一对,这是必须得。看起来就跟C语言里面的函数一样。

第二行对一个特殊的符号 “.” 进行赋值,“.” 在链接脚本里叫做定位计数器,默认的定位计数器为0。我们要求代码链接到 0x10000000 位起始位置的地方,因此这一行给 “.” 赋值 0x10000000 ,表示以 0x10000000 开始,后面的文件或者段都会以 0x10000000 为起始地址开始链接。

第3行的 ".text" 是段名,后面的冒号是语法要求,冒号后面的大括号里面可以填上要链接到 ".text" 这个段里的所有文件,“*(.text)” 中的 "*" 是通配符,表示所有输入文件的 .text 段都放到 ".text" 中。

第四行,我们的要求是将数据放到 0x30000000 开始的地方,所以我们需要重新设置定位计数器,将其改为 0x30000000。如果不重新设置的话会怎样?假设“.text”段的大小为 0x10000,那么接下来的 .data 段的起始几十就是 0x10000000 + 0x10000 = 0x10010000,者明显不符合我们的要求。所以我们必须调整定位计数器为 0x30000000。

第五行跟第三行一行,定义了一个名字位 ".data" 的段,然后所有文件的 ".data" 段都放到这里面。但是这一行多了一个 "ALIGN(4)" ,这是什么意思呢?这是用来对 .data 这个段的起始地址做字节对齐的,ALGN(4) 表示4字节对齐。也就是说段 ".data" 的起始地址要能被 4 整除,一般常见的都是 ALIGN(4) 或者 ALIGN(8),也就是4字节对齐或者8字节对齐。

第6行定义了一个 “.bss” 段,所有文件中的 ".bss" 数据都会北方这个里面,".ss" 数据就是哪些定义了但是没有被初始化的变量。

上面就是链接脚本最基本的语法格式,我们接下来就按照这个基本的语法格式来编写我们本次 “C语言LED灯驱动程序”实验的链接脚本,我们本次实验的链接脚本要求如下:

  1. 链接起始地址为 0x87800000。
  2. start.o 要被链接到最开始的地方,因为 start.o 里面包含了第一个要执行的命令。

根据要求,在Makefle 同目录下创建一个 "im6ul.lds"的文件,然后在文件里输入如下代码:

SECTIONS {. = 0x87800000;.text : {start.omain.o*(.text)}.rodata ALIGN(4) : { *(.rodata*) }.data   ALIGN(4) : { *(.data) }__bss_start = .;.bss    ALIGN(4) : { *(.bss) *(COMMON) }__bss_end = .;
}

上面的链接脚本,其第2行设置定位计数器为 0x87800000 ,因为我们的链接地址就是 0x87800000。第5行设置链接到开始为止的文件为 start.o ,因为 start.o 里面包含着第一个要执行的指令,所以一定要链接到最开始的地方。第6行时 main.o 这个文件,起始这里可以不用写出来,因为 main.o 的位置就无所谓了,可以有由编译器自行决定链接位置。第11,13 行有 "__bss_start" 和 "__bss_end" 这两个东西?这个是什么呢? "__bss_start" 和 "__bss_end" 是符号,第11, 13 这两行起始就是对这两个符号赋值,其值为定位符号 "." ,这两个符号用来保存 .bss 段的起始地址和结束地址。前面说了 .bss 段是定义了但没有被初始化的变量,我们需要手动对 .bss 段的变量清零的,因此我们需要知道 .bss 段的起始地址和结束地址,这样我们直接对这段内存赋 0 即可完成清零。通过第11,13 行代码,.bss 段的起始地址和结束地址就保存在了 "__bss_start" 和 "__bss_end" 中,我们就可以直接在汇编或者C文件里面使用这两个符号了。

3. 修改Makefile使用 "imx6u.lds" 链接脚本

在上一节我们已经编写好了链接脚本 imx6u.lds ,我们可定是要使用这个链接脚本文件的,将Makifile中的如下一行代码:

arm-linux-gnueabihf-ld -Ttext 0x87800000 $^ -o ledc.elf

修改为:

arm-linux-gnueabihf-ld -Timx6ul.lds -o led.elf $^

其实就是将 “-T” 后面的 0x87800000 修改为 imx6ul.lds,表示使用 imx6ul.lds 这个链接脚本文件。修改完成以后使用新的 Makefile 和 链接脚本文件重新编译功能,编译成功之后就可以烧写到 SD 卡验证了。

链接脚本实验中,容易遇到的 .lds 文件的语法错误(我自己实验遇到的):

  • 在赋值语句的后面,缺失了分号 “;”
  • 在 ".bss" 段名语句中间,缺失了冒号 ":"

在编译的时候,编译器发现 .lds 链接脚本的语法错误就会输出错误提示信息,如下图,需要根据编译器提示的错误信息,修正 .lds 链接脚本文件里的语法错误。

(上面这张图就是我自己在第三行 '. = 0x87800000' 这一行尾缺失了分号";",编译其检查到 链接脚本语法错误,输出的错误提示信息。我们根据错误提示信息的行数修正第三行,加上分号。)

4. 下载验证

使用修改后的 Makefile,通过链接脚本控制文件的链接起始为止为 0x87800000,并且通过链接脚本指定文件的链接顺序,把 start.o 文件链接到最终文件的起始位置 0x87800000 处。

imxdownload ledc.bin /dev/sdb

烧录SD卡后,把SD卡查到正点原子 I.MX6ULL APHA/Mini 开发板上,开发板上电验证LED灯是否闪烁。

我验证的结果是使用链接脚本控制 led.elf 的链接位置和文件链接顺序后,使用正点原子提供的 imxdownlaod 烧写到SD卡中后,开发板LED灯正产闪烁。

这篇关于正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-8.2-链接脚本的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/951052

相关文章

10个Python自动化办公的脚本分享

《10个Python自动化办公的脚本分享》在日常办公中,我们常常会被繁琐、重复的任务占据大量时间,本文为大家分享了10个实用的Python自动化办公案例及源码,希望对大家有所帮助... 目录1. 批量处理 Excel 文件2. 自动发送邮件3. 批量重命名文件4. 数据清洗5. 生成 PPT6. 自动化测试

使用Java实现一个解析CURL脚本小工具

《使用Java实现一个解析CURL脚本小工具》文章介绍了如何使用Java实现一个解析CURL脚本的工具,该工具可以将CURL脚本中的Header解析为KVMap结构,获取URL路径、请求类型,解析UR... 目录使用示例实现原理具体实现CurlParserUtilCurlEntityICurlHandler

Linux环境变量&&进程地址空间详解

《Linux环境变量&&进程地址空间详解》本文介绍了Linux环境变量、命令行参数、进程地址空间以及Linux内核进程调度队列的相关知识,环境变量是系统运行环境的参数,命令行参数用于传递给程序的参数,... 目录一、初步认识环境变量1.1常见的环境变量1.2环境变量的基本概念二、命令行参数2.1通过命令编程

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的

Linux编译器--gcc/g++使用方式

《Linux编译器--gcc/g++使用方式》文章主要介绍了C/C++程序的编译过程,包括预编译、编译、汇编和链接四个阶段,并详细解释了每个阶段的作用和具体操作,同时,还介绍了调试和发布版本的概念... 目录一、预编译指令1.1预处理功能1.2指令1.3问题扩展二、编译(生成汇编)三、汇编(生成二进制机器语

10个Python Excel自动化脚本分享

《10个PythonExcel自动化脚本分享》在数据处理和分析的过程中,Excel文件是我们日常工作中常见的格式,本文将分享10个实用的Excel自动化脚本,希望可以帮助大家更轻松地掌握这些技能... 目录1. Excel单元格批量填充2. 设置行高与列宽3. 根据条件删除行4. 创建新的Excel工作表5

Rsnapshot怎么用? 基于Rsync的强大Linux备份工具使用指南

《Rsnapshot怎么用?基于Rsync的强大Linux备份工具使用指南》Rsnapshot不仅可以备份本地文件,还能通过SSH备份远程文件,接下来详细介绍如何安装、配置和使用Rsnaps... Rsnapshot 是一款开源的文件系统快照工具。它结合了 Rsync 和 SSH 的能力,可以帮助你在 li

Linux部署jar包过程

《Linux部署jar包过程》文章介绍了在Linux系统上部署Java(jar)包时需要注意的几个关键点,包括统一JDK版本、添加打包插件、修改数据库密码以及正确执行jar包的方法... 目录linux部署jar包1.统一jdk版本2.打包插件依赖3.修改密码4.执行jar包总结Linux部署jar包部署

mysqld_multi在Linux服务器上运行多个MySQL实例

《mysqld_multi在Linux服务器上运行多个MySQL实例》在Linux系统上使用mysqld_multi来启动和管理多个MySQL实例是一种常见的做法,这种方式允许你在同一台机器上运行多个... 目录1. 安装mysql2. 配置文件示例配置文件3. 创建数据目录4. 启动和管理实例启动所有实例

Linux内存泄露的原因排查和解决方案(内存管理方法)

《Linux内存泄露的原因排查和解决方案(内存管理方法)》文章主要介绍了运维团队在Linux处理LB服务内存暴涨、内存报警问题的过程,从发现问题、排查原因到制定解决方案,并从中学习了Linux内存管理... 目录一、问题二、排查过程三、解决方案四、内存管理方法1)linux内存寻址2)Linux分页机制3)