在新ARM板上移植U-Boot和Linux指南

序言

  • 从支持一个定制板子在U-Boot和Linux中的过程中得到经验
  • 以一个带有知名SoC(i.MX6)且IP已经得到支持的板子为例,这次讨论几乎不涉及编码技能,
  • 更多地聚焦在U-Boot部分

一般原则

  • 如果您有您的BSP(板级支持包)的源代码,请编译并运行BSP以:
    1. 验证您正在工作的IP(知识产权/硬件模块)能够与某些代码一起工作
    2. 拥有一个参考代码
    3. 拥有一个您可以用于调试的代码
  • 专注于正确配置RAM和UART
  • 提交
  • 一次处理一个IP
  • 提交

定制板介绍

  • 基于i.MX6的模块,带有扩展板,
  • 以太网,I2C,SPI,NAND,eMMC,SD卡读卡器,USB设备,EEPROM,
    GPIO,UART,音频(I2S),HDMI,LVDS,PCIe,USB主机,RTC,电源管理集成电路(PMIC)

U-Boot移植

U-Boot 状态

  • 持续从板头文件定义迁移到Kconfig选项,
  • 持续从手动驱动探测迁移到驱动模型

U-Boot 目录结构

  • arch/
    与架构或平台相关的所有内容:设备树(DTS)、CPU初始化、引脚复用控制器、DRAM、时钟等…
  • board/
    特定于板子的代码(初始化、引脚复用配置等),指定板头文件的Kconfig文件,板文件,路径,板文件的Makefile,
  • configs/
    所有板子的默认配置(defconfigs)
  • drivers/
  • include/
    所有头文件
  • include/configs/
    所有板子的头文件

U-Boot新板支持工作流程

  1. 创建板文件
  2. 创建板Kconfig文件
  3. 创建板Makefile
  4. 创建板defconfig
  5. 创建板头文件
  6. 在架构的Kconfig中包含板的Kconfig
  7. 在其CPU的Kconfig中定义TARGET Kconfig选项
    一些平台共享公共文件,因此只需要一个defconfig

创建板文件

board/my_vendor/my_board/my_board.c
在这里插入图片描述

  • 声明全局数据指针
  • 可以在代码中使用全局变量gd
  • 在ARM架构中,对于ARM32等于硬件寄存器r9,对于ARM64等于x18
  • 用于在启动后非常早期的“某些内存”中存储信息,这些内存在系统初始化期间可用(直到我们设置好内存控制器,可以使用RAM)
  • 查看include/asm-generic/global_data.h可以了解它能存储哪些类型的信息。

创建板Kconfig文件

board/my_vendor/my_board/Kconfig

在这里插入图片描述

  • SYS_VENDOR 和 SYS_BOARD 用于标识目录,这样 make 命令能找到编译所需的文件
    - 如果两者都存在
    board/SYS_VENDOR/SYS_BOARD/
    - 如果省略了 SYS_VENDOR
    board/SYS_BOARD/
    - 如果省略了 SYS_BOARD
    board/SYS_VENDOR/common/
  • SYS_CONFIG_NAME 用于标识板头文件
    include/configs/SYS_CONFIG_NAME.h

创建板Makefile文件

board/my_vendor/my_board/Makefile
在这里插入图片描述

创建板defconfig文件

configs/my_board_defconfig

在这里插入图片描述

  • 在这里放置任何可以在Kconfig(菜单配置)中选择的内容
  • 驱动程序、特性、U-Boot行为、库等

创建板header文件

include/configs/my_board.h
在这里插入图片描述

引入板子的Kconfig文件

arch/arm/Kconfig or
arch/arm/mach-imx/mx6/Kconfig
在这里插入图片描述

定义板子的TARGET Kconfig选项

arch/arm/mach-imx/mx6/Kconfig
在这里插入图片描述

U-Boot初始化序列

  • U-Boot会运行两组函数列表,其目的是在用户访问控制台之前初始化或配置特定的IP
  • 第一个列表定义在common/board_f.c文件中的static init_fnc_t init_sequence_f[]数组
  • 第一个列表负责初始化DRAM,映射它,并在工作后重新定位引导程序代码
  • 第二个列表定义在common/board_r.c文件中的static init_fnc_t init_sequence_r[]数组
  • 只有当定义了常量时,才会运行某些函数(例如,定义CONFIG_BOARD_EARLY_INIT_F以运行board_early_init_f())
  • 任何返回非零值的函数都会停止初始化序列,并使U-Boot无法启动
  • 在初始化序列有问题时定义DEBUG
  • 并非所有”特性”在所有函数中都可用(例如,board_early_init_f()中没有udelay)。

驱动选择

  • 从具有相同IP的板子中获取灵感
  • 检查相应子系统中的驱动程序
  1. 专注于驱动程序的行为
  2. 然后查看寄存器、位偏移、掩码等
  3. 检查未定义的宏或常量
  4. 查找被ifdef块包围的代码片段
    - 在子系统的Makefile中查找这个驱动程序的对象文件在这里插入图片描述
  • 使用grep搜索CONFIG_MY_DRIVER
    - 如果在某些Kconfig文件中是可见符号 => 添加到板子的defconfig中
    - 如果在某些Kconfig文件中是非可见符号或未定义 => 添加到板头文件中
  • 确保你的驱动程序被编译(查找my_driver.o)
  • 驱动程序文件位于drivers/mtd/nand/nand_mxs.c
  • 使用CONFIG_NAND_MXS来编译驱动程序
  • 使用CONFIG_SYS_MAX_NAND_DEVICE和CONFIG_SYS_NAND_BASE常量来配置设备

驱动选择 - NAND示例

configs/my_board_defconfig
在这里插入图片描述
include/configs/my_board.h
在这里插入图片描述
board/my_vendor/my_board.c
在这里插入图片描述

关于设备树的说明

  • 设备树的迁移始于2012年,代码正逐渐迁移,逐个驱动程序,逐个子系统
  • 驱动模型需要使用设备树
  • 大多数驱动程序有大量的CONFIG_DM条件编译块
    - 你不能真正基于每个驱动程序选择启用DM支持
  • 子系统核心代码也是如此
  • 我没有深入研究,因为我们要么不需要,要么需要完全支持设备树,而NAND框架还没有迁移到DM。

Linux Kernel 移植

Linux内核新板支持工作流程

1. 创建板子的设备树
2. 将板子的DTB添加到架构DTS Makefile中
3. 为你的板子创建一个默认配置(defconfig)

Device Tree

  • 一个特殊DTS(设备树源)格式的文件
  • 纯粹描述了你的板子的硬件
  • 通过兼容字符串将IP与驱动程序匹配
  • 文档可以在Documentation/devicetree/bindings中找到
  • 有关设备树是什么以及如何编写的更深入解释,请访问:设备树详细介绍视频。

创建板子的设备树

arch/arm/boot/dts/imx6s-my-board.dts
在这里插入图片描述

将DTB添加到架构DTS Makefile中

arch/arm/boot/dts/Makefile
在这里插入图片描述

为你的板子创建一个默认配置(defconfig)

  • 从SoC系列的defconfig开始(例如imx_v6_v7_defconfig),如果没有,从架构的defconfig开始(例如multi_v7_defconfig)
  • 从defconfig中删除无用的SoC系列、驱动程序和特性
  • 添加你想要编译的驱动程序的CONFIG
    - 使用grep搜索驱动程序的基本名称是正确的方法
    - 大多数驱动程序依赖于子系统或其他选项,如果你的驱动程序没有选择它们,你也需要将它们添加到你的defconfig中
  • PCIe驱动程序正在探测但未枚举设备
    - 驱动程序在BSP中工作,发现缺少对电源管理器的支持是罪魁祸首
    - 迅速编写了一个40行的补丁并提交给了上游
  • 以太网驱动程序缺少在BSP中设置的PHY复位后延迟
    - 迅速编写了一个20行的补丁并提交给了上游。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/433817.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【全新课程】正点原子《基于GD32 ARM32单片机项目实战入门》培训课程上线!

正点原子《基于GD32 ARM32单片机项目实战入门》全新培训课程上线啦!正点原子工程师手把手教你学!彻底解决ARM32单片机项目入门难的问题! 一、课程介绍 本课程专为ARM32单片机的入门学习者设计,涵盖了环境搭建、编程软件使用、模…

【数学分析笔记】第3章第4节闭区间上的连续函数(2)

3. 函数极限与连续函数 3.4 闭区间上的连续函数 3.4.4 中间值定理 【定理3.4.4】若 f ( x ) f(x) f(x)在 [ a , b ] [a,b] [a,b]上连续,则它一定能取到最大值 M M M与最小值 m m m之间的任何一个值。 M max ⁡ f ( x ) , x ∈ [ a , b ] , m min ⁡ f ( x ) , …

【前端框架对比和选择】React 与 Vue 框架设计思路对比

框架总览 前端框架繁多,在学习的时候也会陷入困惑,我们应该抓住最主流的内容 Vue/React,深入底层,尝试揣摩框架作者的设计思路,开阔前端培训自己的视野,大家也不要把自己限制在框架之中,认为工…

常见区块链数据模型介绍

除了加密技术和共识算法,区块链技术还依赖于一种数据模型,它决定了信息如何被结构化、验证和存储。数据模型定义了账户如何管理,状态转换如何发生,以及用户和开发者如何与系统交互。 在区块链技术的短暂历史中,数据…

数据治理005-血缘关系

数据血缘是元数据产品的核心能力,但数据血缘是典型的看起来很美好但用起来门槛很高的技术,只要你采买过元数据产品就知道了。这篇文章对数据血缘的特征、价值、用途和方法做了系统阐述: 1、特征:归属性、多源性、可追溯及层次性 2…

SAP已知事务码查询关联角色

运维期间客户就出现没有某些事务码的权限,要求添加; 想要添加事务码就必须知道这个事务码属于哪个角色;使用SUIM-角色-按菜单中的事务分配,输入事务码,点击执行就可以查看 找到相关的角色之后,用SU01添加至…

动态规划算法:12.简单多状态 dp 问题_打家劫舍_C++

目录 题目链接:LCR 089. 打家劫舍 - 力扣(LeetCode) 一、题目解析 题目: 解析: 二、算法原理 1、状态表示 状态表示: 2、状态转移方程 状态转移方程推理: 3、初始化 dp表初始化: 特殊…

【抓包工具】如何下载抓包工具Fiddler

目录 Fiddler简介 Fiddler下载步骤 Fiddler安装步骤 配置Fiddler抓取HTTPS Fiddler简介 Fiddler是一个http协议调试代理工具,它能够记录并检查所有你的电脑和互联网之间的http通讯,设置断点,查看所有的“进出”Fiddler的数据&#xff08…

【BurpSuite】SQL注入 | SQL injection(1-2)

🏘️个人主页: 点燃银河尽头的篝火(●’◡’●) 如果文章有帮到你的话记得点赞👍收藏💗支持一下哦 【BurpSuite】SQL注入 | SQL injection(1-2) 实验一 Lab: SQL injection vulnerability in WHERE clause…

大数据新视界 --大数据大厂之数据压缩算法比较与应用:节省存储空间

💖💖💖亲爱的朋友们,热烈欢迎你们来到 青云交的博客!能与你们在此邂逅,我满心欢喜,深感无比荣幸。在这个瞬息万变的时代,我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

2-105 基于matlab的GA-WNN预测算法

基于matlab的GA-WNN预测算法。遗传算法优化小波神经网络的步骤:1设种群规模为M。随机生成初始种群N , 采用实数编码对个体Ni编码。2、用1中的种群N训练, WNN参数由初始化获得。3、计算种群N中个体适应度值。满足终止条件则跳至6, 不满足执行4。4、适应度大的个体, 选…

携手SelectDB,观测云实现性能与成本的双重飞跃

在刚刚落下帷幕的2024云栖大会上,观测云又一次迎来了全面革新。携手SelectDB,实现了技术的飞跃,这不仅彰显了观测云在监控观测领域的技术实力,也预示着我们可以为全球用户提供更加高效、稳定的数据监测与分析服务。这一技术升级&a…

智慧园区建设,构建智能监控和安防体系

智慧园区是指运用先进的信息技术和互联网思维,以提升园区管理和服务水平为目标,通过整合各类资源、优化园区运营,打造智能化、智能、绿色、低碳的现代园区。在智慧园区中,智慧楼宇、智能监控、智慧消防和智慧安防是不可或缺的重要…

SpringBoot整合JPA实现CRUD详解

SpringBoot版本是2.0以上(2.6.13) JDK是1.8 一、依赖 <dependencies><!-- jdbc --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jdbc</artifactId></dependency><!--…

【ADC】SAR 型 ADC 和 ΔΣ ADC 的选型决策方法

本文学习于TI 高精度实验室课程&#xff0c;介绍如何选择 SAR 或 delta-sigma 型 ADC。 文章目录 一、选型决策树二、特定传感器的应用三、需要 DC 精度但分辨率较低的应用四、需要 DC 精度且分辨率较高的应用五、极低噪声的 DC 精密测量六、需要捕获瞬态信号值的应用七、需要高…

vue单点登录异步执行请求https://xxx.com获取并处理数据

一、请求一个加密地址获取access_token再拼接字符串再次请求 接口返回数据 异步执行请求该地址获取数据并处理 二、请求代码第二步使用 access_token 获取 auth_key // 第二步&#xff1a;使用 access_token 获取 auth_keyconst access_token tokenData.access_token;const …

13年408计算机考研-计算机网络

第一题&#xff1a; 解析&#xff1a;OSI体系结构 OSI参考模型&#xff0c;由下至上依次是&#xff1a;物理层-数据链路层-网络层-运输层-会话层-表示层-应用层。 A.对话管理显然属于会话层&#xff0c; B.数据格式转换&#xff0c;是表示层要解决的问题&#xff0c;很显然答案…

代理模式简介:静态代理VS与动态代理

代理模式&#xff1a;静态代理VS动态代理 1、定义2、分类2.1 静态代理2.2 动态代理 3、使用场景4、总结 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 1、定义 代理模式是一种设计模式&#xff0c;通过代理对象控制对目标对象的访问。简而…

JS---获取浏览器可视窗口的尺寸

innerHeight 和 innerWidth 这两个方法分别是用来获取浏览器窗口的宽度和高度&#xff08;包含滚动条的&#xff09; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible&q…

数据结构 ——— 顺序表oj题:编写函数,删除有序数组中的重复项

目录 题目要求 代码实现 题目要求 一个升序排列的数组 nums &#xff0c;要求原地删除重复出现的元素&#xff0c;使每个元素只出现一次&#xff0c;并返回删除后数组的新长度&#xff0c;元素的相对顺序应该保持一致 代码实现 代码演示&#xff1a; int removeDuplicate…