OpenHarmony 3.0移植到ARM单片机
9 月 30 日,OpenHarmony 3.0 LTS 版本发布。本文将介绍如何移植 OpenHarmony 3.0 到星空派开发板上。
星空派开发板介绍
星空派(GD)开发板是由旗点科技推出的一款 gd32 开发板,板载 GD32F303ZET6 芯片,可直接替代 STM32F103 和 GD32F103 系列。
它支持 WiFi、4G、loRa 等物联通信接口。板载 Flash、eeprom 等,支持 3.2 寸的 TFT-LCD 屏幕。
ARM 芯片移植轻量系统基础知识
在做芯片移植工作之前,我们需要掌握一点点基础知识。
①适配 LiteOS-M 轻量系统
GD32F303 系列器件是基于 Arm® Cortex®-M4 处理器的 32 位通用微控制器。
所以我们使用的是内核是 LiteOS-M,对应的是 OpenHarmony 轻量系统。
可以看到目前已经支持了 cortex-m4 核。所以内核移植工作基本不需要,减少了我们很多工作量,但是我们仍然需要移植 GD32F303 芯片相关的。
②哈佛架构
GD32F303 采用的是哈佛架构,哈佛架构的特点是代码指令和数据分开存储。对于 GD32F303 而言,代码是存放在片内 flash 上,地址是 0x8000000。
③编译结果分析
对于 GD32F303 而言,编译出来的固件一般是 bin 格式、或者 hex 格式。
通常编译最后的结果会产生 4 种不同的内容,有时我们也称为 4 段:
code:即程序代码部分,该内容由所有程序指令组成,也是代码运行的主体,通常是要烧录到 GD32F303 片内 flash 上。
RO-data:只读数据段,例如我们在程序中所定义的全局常量数据和字符串都位于此处。由于这些数据都是只读,不会改变的,那这些只读的全局就没必要放到内存种,可以直接放到 flash 中,可以节省内存。
RW-data:已初始化的读写数据,程序中定义并且初始化的全局变量和静态变量位于此处。由于内存刚上电后,内存上的数据是未知,所以我们需要事先把这些全局变量、静态变量的初始值先存放到 flash 中,然后上电后,由 CPU 将 flash 中的初始值赋予到内存中的变量中。
ZI-data:未初始化的全局变量或者初始化为0的全局变量,这些变量默认都是 0,我们只需要 CPU 上电后,将这些内存都清零即可。
由上我们可以的出来 2 个结论:
2)芯片上电后,GD32F303 需要将 RW-data 的内容复制到内存对应位置,从而保证初始化的全局变量和静态变量的值正确;还要对内存中的 ZI-data 段进行清零操作,最后才能执行 main 函数。
④程序如何启动
对于 ARM Cortex-M 系列的芯片而言,当芯片上电后,ARM 核会将地址为 0x8000000 的数据映射到 0 地址,然后从 0 地址开始读取程序指令。
而 0x8000000 地址是芯片内部 flash 的起始地址。也就是编译生成的固件最终要烧录到的地址。所以我们的固件前面的代码非常重要,它是我们芯片启动后执行的第一条指令。
⑤中断向量表
对于 ARM Cortex-M 系列的芯片而言,0x8000000 地址第一个字节是栈指针,由于栈是从高往下增长的,所以该栈指向芯片最大内存处即可。
接下来从 0x8000000 的第 2 个字节开始是中断向量表,存放着所有中断处理函数指针。
前面 16 个是内核中断,其中第一个中断指针存放的是 Reset_Handler 复位中断处理函数。
移植GD32F303到OpenHarmony 3.0
https://gitee.com/qidiyun/gd32-f303-for-open-harmony-3.0
①先按官网教程搭建好 Ubuntu 下的开发环境
https://gitee.com/harylee/gcc-arm-none-eabi-10-2020-q4-major.git
将交叉编译器环境变量 bin 目录配置到 .bashrc 文件中或者配置 device/st/stm32l4r9i_disco/liteos_m/config.gni 文件中 board_toolchain_path 宏为交叉编译器 bin 路径。
②下载上方移植好的代码
将 device 文件夹下的 gd 文件夹复制到 OpenHarmony 3.0 代码的 device 文件夹下。
将 vendor 文件夹下的 gd 文件夹复制到 OpenHarmony 3.0 代码的 vendor 文件夹下。
③编译
进入 OpenHarmony 3.0 源码根目录,输入 hb set 可以看到由 gd32f303_qidian 的编译选项。选择。
有看到 [OHOS INFO] gd32f303_qidian build success 表示编译成功。
vendor 文件夹
device 代码解析:
由于移植内容较长,本文重点介绍启动文件、链接脚本。
①启动文件
启动文件 startup_gd32f30x.s 。最重要的是将 RW-data 的内容复制到内存对应位置,从而保证初始化的全局变量和静态变量的值正确;还要对内存中的 ZI-data 段进行清零操作,最后才能执行 main 函数。
②链接脚本
链接脚本是“gd32f30x_qidian.ld”,用于指定code、RO-data、RW-data、ZI-data 如何分布。
这里跟芯片相关,需要正确修改,否则可能起不来。
③固件生成规则
接下来就是 text,也就是代码段,还有 rodata,只读数据段。后面还有其它定义,我们下一篇再细节。至此我们的启动文件、链接脚本中比较关键的部分已经说明完。
liteos-m 内核配置
进入“device\gd\gd32f303_qidian\liteos_m”文件夹,可以看到有这个文件“config.gni”,该文件用于配置 liteos-m 内核。
我们使用的交叉编译工具链是 “arm-none-eabi-”。
好了,初步移植要点讲完了,下一篇文章讲 OpenHarmony 内核配置文件、main 函数启动后如何进入鸿蒙轻量内核~
👇扫码报名今晚鸿蒙直播课👇