正点原子[第二期]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

相关文章

基于Linux的ffmpeg python的关键帧抽取

《基于Linux的ffmpegpython的关键帧抽取》本文主要介绍了基于Linux的ffmpegpython的关键帧抽取,实现以按帧或时间间隔抽取关键帧,文中通过示例代码介绍的非常详细,对大家的学... 目录1.FFmpeg的环境配置1) 创建一个虚拟环境envjavascript2) ffmpeg-py

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

Linux链表操作方式

《Linux链表操作方式》:本文主要介绍Linux链表操作方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、链表基础概念与内核链表优势二、内核链表结构与宏解析三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势六、典型应用场景七、调试技巧与

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

详解Linux中常见环境变量的特点与设置

《详解Linux中常见环境变量的特点与设置》环境变量是操作系统和用户设置的一些动态键值对,为运行的程序提供配置信息,理解环境变量对于系统管理、软件开发都很重要,下面小编就为大家详细介绍一下吧... 目录前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

Linux中修改Apache HTTP Server(httpd)默认端口的完整指南

《Linux中修改ApacheHTTPServer(httpd)默认端口的完整指南》ApacheHTTPServer(简称httpd)是Linux系统中最常用的Web服务器之一,本文将详细介绍如何... 目录一、修改 httpd 默认端口的步骤1. 查找 httpd 配置文件路径2. 编辑配置文件3. 保存

Linux使用scp进行远程目录文件复制的详细步骤和示例

《Linux使用scp进行远程目录文件复制的详细步骤和示例》在Linux系统中,scp(安全复制协议)是一个使用SSH(安全外壳协议)进行文件和目录安全传输的命令,它允许在远程主机之间复制文件和目录,... 目录1. 什么是scp?2. 语法3. 示例示例 1: 复制本地目录到远程主机示例 2: 复制远程主

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio