sentry_gimbal_hzz/VSCode+Ozone使用方法.md

71 KiB
Raw Blame History

VSCode+Ozone开发STM32的方法

neozng1@hnu.edu.cn

[TOC]

TODO

  1. 添加一键编译+启用ozone调试脚本使得整个进一步流程自动化
  2. 增加更多的背景知识介绍

前言

了解过嵌入式开发的你一定接触过Keil这款20世纪风格UI的IDE伴随很多人度过了学习单片机的岁月。然而由于其缺少代码补全、高亮和静态检查的支持以及为人诟病的一系列逆天的设置、极慢的编译速度特别是在开发HAL库时很多开发者开始转向其他IDE。

IAR、CubeIDE等都是广为使用的“其他”IDE但是他们也有各自的缺点不能让笔者满意。作为IDE界的艺术家JetBrains推出的Clion也在相当程度上完善了对嵌入式开发的支持。不过在体验过多款IDE后还是VSCode这款高度定制化的编辑器最让人满意。强大的补全和snippet以及代码高亮、定义跳转甩KEIL十条街。

而Ozone则是SEGGER(做jilnk的)推出的调试应用支持变量实时更新变量曲线可视化SEGGER RTT日志DBG虚拟串口等功能大大扩展了调试的功能。很多人习惯使用串口进行可视化调试如vofa串口调试助手等。然而通过这些方式进行调试都是对内核有侵入性会占有内核资源并且导致定时器的时间错乱。由于DBG有单独连接到FLASH和CPU寄存器的高速总线类似于DMA可以在不影响程序正常运行的情况下以极高的频率直接获取变量值。

下面将从工具链介绍、环境配置以及调试工作流三个方面介绍以VSCode为编辑器Ozone为调试接口的开发环境。

开发的大致流程为:

graph LR
CubeMX进行初始化 --> VSCode编写代/进行编译/简单调试 --> Ozone变量可视化调试+log

本教程不仅希望教会你如何配置环境,同样会告诉你每一步究竟是在做什么,而不是简单的复制黏贴邯郸学步。

前置知识

  1. 计算机速成课:Crash Course Computer Science

  2. 从零到一打造一台计算机:

    编程前你最好了解的基本硬件和计算机基础知识(模拟电路)

    编程前你最好了解的基本硬件和计算机基础知识(数字电路)

    从0到1设计一台计算机

  3. C语言基础程序设计入门——C语言

务必学完以上课程再开始本教程的学习。

如果有可能,还应该学习:哈佛大学公开课计算机科学cs50。你将会对单片机和计算机有不同的理解。

预备知识

  1. 软件安装(队伍NAS和资料硬盘内提供了所有必要的依赖,安装包和插件,目录是/EC/VSCode+Ozone环境配置)请以公共账号登陆网盘ip地址为49.123.113.2:5212,账号public@rm.cloud,密码public

    所有安装包也可以在此百度网盘链接下获得:archive.zip

    # 网盘中的文件:
    basic_framework.zip    # 本仓库文件,注意,可能不为最新,建议从仓库clone并定时pull
    daplink_register_license.rar # daplink license注册机
    gcc-arm-none-eabi-10.3-2021.10-win32.zip # arm-gnu-toolchain
    JLinkARM.dll # 修改过的jlink运行链接库
    JLink_Windows_V722b.exe  # JLink软件包
    mingw-get-setup.exe  # mingw工具链
    OpenOCD.zip  # OpenOCD
    Ozone_doc.pdf  # Ozone使用手册
    Ozone_Windows_V324_x86.exe  # Ozone安装包
    VSCodeUserSetup-x64-1.73.1.exe  # VSCode安装包
    
  2. C语言从源代码到.bin和.hex等机器代码的编译和链接过程

  3. C语言的内存模型

  4. C语言标准动态链接库和静态编译的区别一些编译器的常用选项

  5. STM32F4系列的DBG外设工作原理

  6. GDB调试器、硬件调试器和DBG的关系

编译全过程

C语言代码由固定的词汇关键字按照固定的格式语法组织起来简单直观程序员容易识别和理解但是CPU只能识别二进制形式的指令并且这些指令是和硬件相关的感兴趣的同学可以搜索指令集相关内容。这就需要一个工具将C语言代码转换成CPU能够识别的二进制指令对于我们的x86平台windows下的程序就是.exe后缀的文件对于单片机一般来说是.bin或.hex等格式的文件调试文件包括axf和elf

能够完成这个转化过程的工具是一个特殊的软件,叫做编译器Compiler。常见的编译器包括开源的GNU GCCwindows下微软开发的visual C++以及apple主导的llvm/clang。编译器能够识别代码中的关键字、表达式以及各种特定的格式并将他们转换成特定的符号也就是汇编语言(再次注意汇编语言是平台特定的),这个过程称为编译Compile

对于单个.c文件从C语言开始到单片机可识别的.bin文件一般要经历以下几步

img

首先是编译预处理Preprocessing这一步会展开宏并删除注释将多余的空格去除。预处理之后会生成.i文件。

然后,开始编译Compilation的工作。编译器会将源代码进行语法分析、词法分析、语义分析等根据编译设置进行性能优化然后生成汇编代码.s文件。汇编代码仍然是以助记符的形式记录的文本比如将某个地址的数据加载到CPU寄存器等还需要进一步翻译成二进制代码。

下一步就是进行汇编Assemble编译器会根据汇编助记符和机器代码的查找表将所有符号进行替换生成.o .obj等文件。但请注意这些文件并不能直接使用烧录我们在编写代码的时候都会包含一些,因此编译结果应当有多个.o文件。我们还需要一种方法将这些目标文件缝合在一起使得在遇到函数调用的时候程序可以正确地跳转到对应的地方执行。

最后一步就由链接器Linker也称LD完成称为链接Linking。比如你编写了一个motor.c文件和.h文件并在main.c中包含了motor.h使用了后者提供的MotorControl()函数。那么,链接器会根据编译器生成.obj文件时留下的函数入口地址将main.o里的调用映射到生成的motor.o中。链接完成后就生成了单片机可以识别的可执行文件通过支持的串口或下载器烧录便可以运行。

另外,上图可以看到左侧的静态库,包括.lib .a比如我们在STM32中使用的DSP运算库就是这种文件。他在本质上和.o文件相同只要你在你编写的源文件中包含了这些库的头文件链接器就可以根据映射关系找到头文件中声明的函数在库文件的地址。直接提供库而不是.c文件就可以防止源代码泄露因此一些不开源的程序会提供函数调用的头文件和接口具体实现的库你也可以编写自己的库感兴趣自行搜索

链接之后,实际上还要进行不同代码片段的重组、地址重映射,详细的内容请参看:C/C++语言编译链接过程这篇教程还提供了以GCC为例的代码编译示例。

C语言内存模型

image-20221112160213066

以上是C语言常见的内存模型即C语言的代码块以及运行时使用的内存包括函数、变量等的组织方式。

有些平台的图与此相反,栈在最下面(内存低地址),其他区域都倒置,不影响我们理解

代码段即我们编写的代码,也就是前面说的编译和链接之后最终生成的可执行文件占据的空间。一些常量,包括字符串和使用const关键字修饰的变量被放在常量存储区。static修饰的静态变量(包括函数静态变量和文件静态变量)以及全局变量放在常量区上面一点的全局区(也称静态区)。

然后就是最重要的。在一个代码块内定义的变量会被放在栈区,一旦离开作用域(出了它被定义的{}的区域),就会立刻被销毁。在调用函数或进入一个用户自定义的{}都会在栈上开辟一块新的空间空间的大小和内存分配由操作系统或C库自动管理。一般来说,直接通过变量访问栈内存,速度最快(对于单片机)。而堆则是存储程序员自行分配的变量的地方,即使用malloc(),realloc() ,new等方法获取的空间,都被分配在这里。

在CubeMX初始化的时候Project mananger标签页下有一个Linker Setting的选项这里是设置最小堆内存和栈内存的地方。如果你的程序里写了大规模的数组或使用malloc()等分配了大量的空间可能出现栈溢出或堆挤占栈空间的情况。需要根据MCU的资源大小设置合适的stack size和heap size。

RTOS创建任务的时候也会为每个任务分配一定的栈空间它会替代MCU的硬件裸机进行内存的分配。可以在CubeMX中设置。如果一个任务里定义了大量的变量可能导致实时系统运行异常请增大栈空间。

开发板C型使用F407IG芯片片上RAM的大小为1MB。

C language标准和编译器

不同的C语言标准一般以年份作代号支持的语法特性和关键字不同拥有的功能也不同。一般来说语言标准都是向前兼容的在更新之后仍然会保存前代的基本功能支持legacy support。不过为了程序能够正常运行我们还需要一些硬件或平台支持的组件。比如malloc()这个函数在linux平台和windows平台上的具体实现就相去甚远跟单片机更是差了不止一点。前两者一般和对应的操作系统有关后者在裸机上则是直接通过硬件或ST公司提供的硬件抽象层代码实现。

然而不同编译器提供的代码实现也不尽相同比如使用clang和gcc这两种c语言编译器他们对于一些标准库也称C库包括stdiostdlibstring等在内的实现的函数的实现就不太一样。再如__packed是arm-cc提供的一个字节不对齐关键字在一些其他编译器中就不支持这种实现。

以前大家常用的KEIL使用的是ARM提供的arm-cc工具链非常蛋疼甚至不支持uint8_t=0b00001111这种二进制定义法而该教程选用的是开源的Arm GNU Toolchain。在非目标机且和目标机平台不同的平台上进行开发被成为跨平台开发,进行的编译也被成为交叉编译(在一个平台上生成另一个平台上的 可执行代码)。

工具链包含了编译器链接器以及调试器等开发常用组件。我们使用的Arm GNU toolchain中编译器是arm-none-eabi-gcc.exe,链接器是arm-none-eabi-ld.exe,调试器则是arm-none-eabi-gdb.exe。通过跨平台调试器和j-link/st-link/dap-link我们就可以在自己的电脑上对异构平台即单片机的运行进行调试了。

==特别注意在新框架中我们使用的是arm-none-eabi-gcc此编译器不支持__packed关键字,若要进行字节压缩(不对齐字节),应该使用预编译指令#pragma pack(1)==。

Debug外设工作原理

image-20221112145717063

DBG支持模块红框标注部分也可以看作一个外设通过一条专用的AHB-AP总线和调试接口相连Jtag或swd并且有与数据外设总线直接相连的桥接器。它还同时连接了中断嵌套管理器因此同样可以捕获中断并进行debug和ITM、DWT、FPB这些调试支持模块。因此DBG可以直接获取内存或片上外设内的数据而不需要占用CPU的资源并将这些数据通过专用外设总线发送给调试器进而在上位机中读取。

FPB是flash patch breakpoint闪存指令断点的缩写用于提供代码断点插入的支持当CPU的指令寄存器读取到某一条指令时FPB会监测到它的动作并通知TPIU暂停CPU进行现场保护。

DWT是data watch trace数据观察与追踪单元的缩写用于比较debug变量的大小并追踪变量值的变化。当你设定了比较断点规则当某个数据大于/小于某个值时暂停程序或将变量加入watch进行查看DWT就会开始工作。DWT还提供了一个额外的计时器即所有可见的TIM资源之外的另一个硬件计时器因为调试其他硬件定时器的计时由于时钟变化可能定时不准而DWT定时器是始终正常运行的。它用于给自身和其他调试器模块产生的信息打上时间戳。我们的bsp中也封装了dwt计时器你可以使用它来计时。

ITM是instrument trace macrocell指令追踪宏单元的缩写它用于提供非阻塞式的日志发送支持相当于大家常用的串口调试SEGGER RTT就可以利用这个模块向上位机发送日志和信息。这个硬件还可以追踪CPU执行的所有指令这也被称作trace跟踪并将执行过的指令全部通过调试器发送给上位机。当debug无法定位bug所在的时候逐条查看cpu执行的指令是一个绝佳的办法特别是你有大量的中断或开启了实时系统时。

以上三个模块都需要通过TPIUtrace port interface unit和外部调试器j-link等进行连接TPIU会将三个模块发来的数据进行封装并通过DWT记录时间发送给上位机。

GDB调试MCU原理

image-20221117121323757

不论使用MDKKEIL还是VSCode还是Ozone实际上背后的流程相同。首先GDB会建立TCP/IP端口并提供接口调试服务器Server作为硬件调试器和GDB软件的桥梁将硬件调试器的相关功能也就是DBG外设支持的那些功能映射到GDB的接口上通过连接到GDB建立的端口。之后启动调试将可执行文件下载到目标MCU上然后从main开始执行

当然你也可以选择从其他启动点开始执行,调试器开始执行的位置叫做entry point。同样在MCU已经正在运行程序的时候可以attach到程序上开始监控attach=附加,贴上;很形象了)。

而对于直接运行在电脑上的程序(.exe就不需要GDBserver和物理调试器GDB程序可以直接访问电脑上运行的程序和CPU的寄存器等。

字节对齐

这是内存硬件设计和汇编语言设计的结果。在使用结构体的时候如果你不做任何事情编译器会自动帮助你完成字节对齐以提高内存访问的效率。stm32是4字节地址和数据总线的设计单词可以传输32位数据因此访问4字节数据也就是stm32的“字”“字长”效率最高。然而历史的缘由导致一个内存地址只存储8位的数据如果你要访问float数据则一次需要读取4个地址。当这四个地址是连续的时候你只需要一次就可以将数据读出。然而如果一个float数据被存放在0x03-0x07这四个地址cpu首先要读出0x00-0x03这四个连续的地址然后再取出最后一个字节随后读取0x04-0x07这四个连续的字节再取出前三个字节最后将最后一个字节和前三个字节拼接在一起形成我们需要的float数据。

#pragma可能是最复杂的预编译指令,不同的编译器支持不同的#pragma指令,如常用的#pragma once可以替代header guard。arm gnu gcc编译器支持通过#pragma pack()来设置字节对齐,支持的对齐参数包括空/1/2/4/8会启动对应长度的对齐方式。用于通信的结构体串口/CAN/spi等外设接收数据的时候都是连续的不会像结构体一样被编译器对齐在声明时采用如下的方式

#pragma pack(1) // 从这句话开始,使用字节对齐(1),即紧凑,关闭对齐

typedef struct
{
    uint8_t id;
    // ...
    
} CANInstance;

#pragma pack() // 从这里开始,恢复默认配置,一般来说默认配置是 pack(4),如果遇到double/longlong等也会变为8字节对齐
// 使用两个#pragma包裹你的结构体声明

如果您有兴趣,可以了解一下内存硬件的组织和连接方式,包括奇偶地址/片选/行列扩展等,可以帮助你更好地理解字节对齐。

环境配置

所有需要编辑的配置文件都已经在basic_framework的仓库中提供如果不会写照猫画虎。

2022-12-01更新

VSCode上线了一款新的插件

image-20221201134906999

支持一键配置Arm GNU工具链、MinGW64make工具和OpenOCD可以尝试使用这个插件替代下面的配置流程。并且此插件还提供了一键下载、一键调试的支持只需要选择合适的下载器配置即可全部都是图形化界面的操作

你可以尝试使用这个插件进行环境的配置。当然,环境变量仍然需要手动添加。

==另外如果你不想配置太多东西也不想了解底层的信息可以尝试Embedded IDE插件。它支持直接在VSCode中编辑KEIL MDK的项目相关信息请自行查阅或直接查看EIDE的项目网站==https://em-ide.com

  • 安装STM32CubeMX并安装F4支持包和DSP库支持包

  • 安装VSCode并安装以下插件

    • C/C++提供C/C++的调试和代码高亮支持
    • Better C++ Syntax:提供更丰富的代码高亮和智能提示
    • C/C++ Snippets:提供代码块(关键字)补全
    • Cortex-DebugCortex-Debug: Device Support Pack - STM32F4提供调试支持。cortex debug还会自动帮助你安装一些调试相关的插件。
    • IntelliCodeMakfile Tools:提供代码高亮支持

    image-20221112172157533

    image-20221112172208749

    image-20221112172221756

    image-20221112172239386

    image-20221112172254809

    MinGW、Arm GNU toolchain和OpenOCD也可以通过MSYS2使用pacman包管理器和apt/yum类似直接安装这种方法一步到位==这是更推荐使用的方式==,请参看附录5

    **当然,你也可以直接按照下面的方法安装这两个工具。**不过强烈推荐使用附录5中的方法。下面的方法将在发布完整版更新的时候被删除。

  • 安装MinGW等待界面如下will be deprecated soon请注意这种方法将会在主分支发布正式版的时候删除

    image-20221112172051589

    安装好后打开MinGW后将所有的支持包勾选然后安装

    image-20221112172348408

    image-20221112172420037

    安装完以后将MinGW的bin文件夹添加到环境变量中的path下按下菜单键搜索编辑系统环境变量打开之后:

    image-20221112172716320

    图片看不清请打开原图。验证安装:

    打开命令行win+Rcmd回车输入gcc -v,如果没有报错,并输出了一堆路径和参数说明安装成功。

    安装完之后建议将ming的bin文件夹下的ming32-make.exe复制一份并将copy更名为make.exe

  • 配置gcc-arm-none-eabi环境变量把压缩包解压以后放在某个地方然后同上将工具链的bin添加到PATHwill be deprecated soon请注意这种方法将会在主分支发布正式版的时候删除

    image-20221112172858593

    安装路径可能不一样,这里要使用你自己的路径而不是直接抄

    验证安装:

    打开命令行,输入arm-none-eabi-gcc -v,如果没有报错,并输出了一堆路径和参数说明安装成功。

添加到环境变量PATH的意思是当一些程序需要某些依赖或者要打开某些程序时系统会自动前往PATH下寻找对应项。一般需要重启使环境变量生效。

若你不希望扰乱系统的环境变量可以参照附录5将Msys2/MinGW64的终端集成到VSCode中方便开发

  • 将OpenOCD解压到一个文件夹里稍后需要在VSCode的插件中设置这个路径。will be deprecated soon请注意这种方法将会在主分支发布正式版的时候删除

  • CubeMX生成代码

    在project manager标签页工具链选择makefile

    image-20221112173534670

    生成的目录结构如下:

    image-20221112174211802

    Makefile就是我们要使用的构建规则文件。

    如果你使用basic_framework不需要重新生成代码。

  • 建议将ozone和jlink的目录一同加入环境变量方便我们后续的一键下载和一键调试配置

VSCode编译和调试配置

VSCode常用快捷键包括

功能 快捷键
选中当前行 Ctrl+L
删除当前行 Ctrl+Shift+K
重命名变量 F2
跳转到定义 Ctrl+点击
在打开的文件页中切换 Ctrl+Tab
在当前文件查找 Ctrl+F
在整个项目文件夹中查找 Ctrl+Shift+F
查找所有引用 Alt+Shift+F12
返回上一动作 Alt+左

更多快捷键可以按ctrl+K再按ctrl+S显示并且可以修改成你最习惯的方式。此外使用Snippets可以大幅度提高重复性的代码编写速度它可以直接帮你补全一个代码块如for、while、switch补全和snippet都使用Tab键接受代码提示的提议,通过↑和↓键切换提示。

编译

为了提供完整的代码高亮支持需要配置Makefile tools插件的make程序路径ctrl+,打开设置搜索make path找到设置并填写

image-20221113152513343

mingw32-make就是下面介绍的make工具配合makefile替代手动调用gcc。这里之所以只要输入mingw32-make而不用完整路径是因为我们将mingw的bin文件夹加入环境变量了因此系统会在PATH下自动寻找对应项

用VSCode打开创建的项目文件夹Makefile Tools插件会询问你是否帮助配置intellisense选择是。

此时就可以享受intellicode带来的各种便利的功能了。我们的项目使用Makefile进行编译在之前的编译介绍中以GCC编译器为例如果需要编译一个文件要输入如下命令

gcc your_source_code_name.c -o output  # your_source_code_name是待编译的文件名

然而,你面对的是一个拥有几百个.c和.h文件以及大量的链接库如果要将所有文件都输入进去那将是一件苦恼的事。Makefile在gcc命令上提供了一层抽象通过编写makefile来指定参与编译的文件和编译选项再使用make命令进行编译它会自动将makefile的内容“翻译”为gcc命令。这样编译大型项目就不是一件困难的事了。更多关于makefile的指令介绍参见附录3

实际上在使用keil MDK开发的时候它调用的仍然是底层的arm cc工具链中的编译器和链接器在配置“魔术棒”添加项目文件以及包含目录的时候实际做的使其和makefile差不多。keil使用的参数可以在魔棒的C/C++选项卡下看到。

对于一个已经拥有makefile的项目打开一个终端输入

mingw32-make -j24 # -j参数表示参与编译的线程数,一般使用-j12

注意多线程编译的时候输出的报错信息有时候可能会被打乱多个线程同时往一个terminal写入程序运行的信息要是看不清报错请使用mingw32-make,不要进行多线程编译。

我对make的编译命令进行了静默处理只输出error和warning以及最后的生成文件信息。如果想要解除静默就是下面所说的“你可以看到大致如下的输出”需要修改Makefile。本仓库下的makefile中已经用注释标明。

image-20221112191712534

就会开始编译了。你可以看到大致如下的输出:

arm-none-eabi-gcc -c -mcpu=cortex-m4 -mthumb -mfpu=fpv4-sp-d16 -mfloat-abi=hard -DUSE_HAL_DRIVER -DSTM32F407xx -DARM_MATH_CM4 -DARM_MATH_MATRIX_CHECK -DARM_MATH_ROUNDING -IHAL_N_Middlewares/Inc -IHAL_N_Middlewares/Drivers/STM32F4xx_HAL_Driver/Inc -IHAL_N_Middlewares/Drivers/STM32F4xx_HAL_Driver/Inc/Legacy -IHAL_N_Middlewares/Drivers/CMSIS/Device/ST/STM32F4xx/Include -IHAL_N_Middlewares/Drivers/CMSIS/Include -IHAL_N_Middlewares/Drivers/CMSIS/DSP/Include -IHAL_N_Middlewares/Middlewares/ST/STM32_USB_Device_Library/Core/Inc -IHAL_N_Middlewares/Middlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc -IHAL_N_Middlewares/Middlewares/Third_Party/FreeRTOS/Source/CMSIS_RTOS -IHAL_N_Middlewares/Middlewares/Third_Party/FreeRTOS/Source/portable/GCC/ARM_CM4F -IHAL_N_Middlewares/Middlewares/Third_Party/FreeRTOS/Source/include -IHAL_N_Middlewares/Middlewares/Third_Party/FreeRTOS/Source/include -IHAL_N_Middlewares/Middlewares/Third_Party/SEGGER/RTT -IHAL_N_Middlewares/Middlewares/Third_Party/SEGGER/Config -IHAL_N_Middlewares/Middlewares/ST/ARM/DSP/Inc -Iapplication -Ibsp -Imodules/algorithm -Imodules/imu -Imodules/led_light -Imodules/master_machine -Imodules/motor -Imodules/referee -Imodules/remote -Imodules/super_cap  -Og -Wall -fdata-sections -ffunction-sections -g -gdwarf-2 -MMD -MP -MF"build/stm32f4xx_hal_pwr_ex.d" -Wa,-a,-ad,-alms=build/stm32f4xx_hal_pwr_ex.lst HAL_N_Middlewares/Drivers/STM32F4xx_HAL_Driver/Src/stm32f4xx_hal_pwr_ex.c -o build/stm32f4xx_hal_pwr_ex.o

仔细看你会发现make命令根据makefile的内容调用arm-none-eabi-gcc编译器传入了一堆的参数以及编译选项然后运行。

最后输出的结果如下:

  text    data     bss     dec     hex filename
  31100     484   35916   67500   107ac build/basic_framework.elf
arm-none-eabi-objcopy -O ihex build/basic_framework.elf build/basic_framework.hex
arm-none-eabi-objcopy -O binary -S build/basic_framework.elf build/basic_framework.bin

由于使用了多线程编译比KEIL的蜗牛单线程要快了不少。以上内容代表了生成的可执行文件的大小以及格式和内容。.elf文件就是我们需要传递给调试器的东西使用VSCode调试部分会介绍。典型的编译时间大致如下:

  1. 从零开始重新编译:~10s
  2. 修改文件后编译:~3s

当然了,你可能觉得每次编译都要在命令行里输入参数,太麻烦了。我们可以编写一个task.json这是VSCode的一个任务配置内容大致如下

{
    // See https://go.microsoft.com/fwlink/?LinkId=733558
    "version": "2.0.0",
    "tasks": [
        {
            "label": "build task",         // 任务标签
            "type": "shell",               // 任务类型,因为要调用mingw32-make,是在终端(CMD)里运行的,所以是shell任务
            "command": "mingw32-make -j24",// 要执行的任务命令
            "problemMatcher": [],          
            "group": {
                "kind": "build",
                "isDefault": true
            }
        }
    ]
}

这样你就可以点击VSCode工具栏上方的Terminal->Run task选择你刚刚配置的任务开始编译了。更方便的方法是使用快捷键:ctrl+shift+B 之后要配置下载任务和调试任务等也可以利用这种方法新建一个xxx_task实现一键下载、一键调试等。

image-20221112192133103

还没配置任务的时候需要在Terminal标签页中选择Configure Tasks... 创建一个新的.json文件。

P.S. VSCode中的大部分配置都是通过json文件保存的。当前工作区的配置在项目文件夹中的.vscode下全局配置在设置中修改。全局配置在当前工作区没有配置的时候会生效反之被前者覆盖。

如果你编写了新的代码文件

Makefile的大部分内容在CubeMX初始化的时候就会帮你生成。如果新增了.c的源文件你需要在C_SOURCES中新增:

image-20221112192509718

换行需要在行尾加反斜杠\

如果新增了头文件,在C_INCLUDES中新增头文件所在的文件夹:

image-20221112192610543

换行需要在行尾加反斜杠\

添加完之后,重新编译即可

和KEIL新增文件的方式很相似但是更方便。

  • 另外如果你使用的时linux/Unix like/MacOS则可以直接使用根目录下的Makefile.upgrade复制替换到Makefile中我们在其中定义了递归添加源文件和头文件目录的规则不再需要手动添加新增的源文件和头文件路径。如果你使用windows+mingw/Msys2则需要在mingw环境下执行编译指令否则报错因为makefile中使用了一些shell指令是cmd和powershell不支持的后续考虑在makefile中添加os判断规则以自动替换目录查找指令。若你坚持使用cmd/powershell请参照Makefile.upgrade中的注释将makefile修改为对应指令格式以支持该环境下的使用。

简单的调试配置

在VSCode中调试不能像Keil一样查看变量动态变化但是支持以外的所有操作如查看外设和反汇编代码设置断点触发方式等。

用于调试的配置参考这篇博客Cortex-debug 调试器使用介绍,这里包含了一些背景知识的介绍。你也可以直接查看下面的教程。

***==注意==***

如果你的用户名是中文,请先按照附录6将自己的用户名改成英文。

你需要配置arm gnu工具链的路径(工具链包括编译器、链接器和调试器等),OpenOCD的路径使得GDB调试器可以找到OpenOCD并调用它从而连接硬件调试器如j-link等JlinkGDBServer的路径,以及该工作区(文件夹)的launch.json文件用于启动vscode的调试任务

VSCode ctrl+,进入设置,通过搜索找到cortex-debug插件的设置。

  1. 搜索armToolchainPath设置你的arm gcc toolchain的bin文件夹。bin是binary的缩写实际上文件夹内部是一些可执行文件整个工具链都在这里注意该文件夹是刚刚解压的arm gcc toolchain的根目录下的bin文件夹里面有很多以arm-none-eabi为前缀的可执行文件)。此路径必须配置。
  2. 搜索openocdPath设置你的openocd路径需要包含到openocd的可执行文件。使用daplink调试需要配置这个路径。
  3. 搜索JLinkGBDServer设置JlinkGDBServerlCL.exe的路径在Jlink安装目录下CL代表command line命令行版本。使用jlink调试需要配置这个路径。

注意windows下路径需要使用两个反斜杠\\代表下一级文件夹。

如果你使用附录5中的方法安装前两个的路径都在Msys2/mingw64/bin下。

其他配置需要的文件已经全部在basic_framework中提供,包括openocd.cfg STM32F407.svd .vscode/launch.json

image-20221115215531879

主要需要配置这三个路径第四个gdbPath可以选配

如果教程中的启动json文件看不懂请看仓库里的.vscode下的launch.json,照葫芦画瓢。注意把我写的路径替换掉或注释掉。launch.json已经添加了详细的注释。

根目录下已经提供了C板所需的.svd和使用无线调试器时所用的openocd.cfg配置文件。

然后选择run and debug标签页在选项中选择你配置好的选项开始调试。或者使用快捷键:F5

image-20221112180103750

我们的仓库中默认提供了两种下载器的支持dap-link无线调试器属于这一种和j-link包括小的j-link OB和黑色大盒子jlink

调试介绍

开始调试后,显示的界面如下:

  1. 变量查看窗口包括当前调用栈当前作用域或代码块内的局部变量、当前文件的静态变量和全局变量。register选项卡可以查看cpu内核的寄存器数值。

  2. 变量watch窗口。右键单击要查看的变量选择watch加入查看。

    image-20221113131044191

    还支持直接运行到指针所选处Run to Cursor以及直接跳转到指针处执行Jump to Cursor。添加行内断点若一个表达式由多个表达式组成也是很方便的功能可以帮助进一步定位bug。

    右键点击添加到watch窗口的变量**可以临时修改它们的值。**调参的时候非常好用。

    VSCode提供的一个最大的便利就是你可以将鼠标悬停在需要查看的变量上**不需要添加到watch就能观察变量值。**如果是指针还可以自动解析,获取解引用后的值。结构体也支持直接展开。

    image-20221113133624273

    现在Cortex-Debug插件也已经支持live watch变量动态监视最高可设置的刷新频率为4Hz足堪大用我们可以宣告KEIL的时代已经落幕但是更复杂更高频率的变量观测以及可视化功能还是需要通过ozone完成。

  3. 调用栈。表明在进入当前代码块之前调用了哪些函数,称之为栈也是因为调用的顺序从下至上。当前函数结束之后栈指针会减小,控制权会返还给上一级的调用者。通过调用栈可以确认程序是如何(按怎样的顺序)运行到当前位置的。

  4. 片上外设。这里可以查看外设的控制寄存器状态寄存器的值如果通过断点无法定位bug则需要查找数据手册和Cortex M4指南的相关内容根据寄存器值来判断程序当前的情况。

  5. 断点。所有添加的断点都会显示于此注意不像我们自己的电脑单片机的DBG外设对断点的数量有限制资源所限超过5个断点会导致debug失败此时将断点减少即可。由于单行代码编译之后可能会对应多条汇编指令或一条表达式由多个表达式构成你还可以插入行内断点以逐个执行表达式或汇编语句,你还可以在汇编窗口插入断点调试汇编代码,帮助你发现错误。

    对于不方便判断何时需要停止代码执行进行观察和测试的情况,你可以右键行号左侧的断点栏插入条件断点,输入表达式,当表达式满足时才会进入断点。

  6. 调试控制台。调试器输出的信息会显示在这里,要查看追踪的变量的信息也会显示在这里。如果调试出现问题报错信息同样也会在这里显示。要是出现异常可以复制这里的信息在搜索引擎里查找答案不过最好的方法是查询gdb和openocd的官方文档。

  7. 调试控制。

    • 复位:单片机复位
    • 继续运行/暂停
    • 单步跳过,如果这一行有函数调用,不会进入内部
    • 进入,如果这一行有函数调用,会进入函数内部
    • 跳出跳出当前调用栈顶层的函数即如果在函数内部会直接运行到return
    • 重启调试器(当然单片机也会复位,一般出现异常的时候使用这个按钮)
    • 终止调试

如果你希望在编译之后立刻启动调试,不要分两次点击,你可以在launch.json中添加一个prelaunchtask(意为在启动调试之前要运行的任务),将他设置为我们在编译章节介绍的构建任务。我们已经提供了这个选项,取消注释即可使用。

如果你想在VSCode中也使用segger RTT viewer的功能即bsp_log提供的日志功能请参阅[附录2](##附录2在VSCode中启用SEGGER RTT日志)。

如果想直接下载代码不想调试,参阅附录4

RTT Viewer日志功能

2023/07/23补充Cortex-Debug插件最近似乎集成了RTT Viewer的支持只需要设置好RTT client的路径便可以一键启动RTT终端查看串行调试器中发送的内容。还支持以一定的格式将RTT发来的内容进行可视化示波器但支持程度不如Ozone。可以自行查看插件的wiki和文档进行配置。

本框架添加了vscode下Segger RTT client的支持。在.vscode/task.json中已经添加了启动rtt viewer client的任务。你也可以将此任务作为附加启动任务和调试一起启动方便查看日志。要使用日志请包含bsp_log.h。注意需要将jlink的安装目录添加到环境变量中。

更好的编辑体验

建议安装以下插件:

  1. Hex Hover Converter鼠标悬停在数值上的时候会自动显示其对应的16、2、10进制值和编码

  2. Hex Editor在查看汇编代码和机器代码的时候提供2、10、16进制转换并且可以以16进制或2进制的格式编辑文件。

  3. GitLens和git graph提供强大的可视化commit记录和UI支持

  4. Blockman - Highlight Nested Code Blocks 此插件会高亮嵌套的代码块即花括号包围的部分或for/while/ifelse代码块对于多层条件和循环嵌套效果非常炸裂

  5. bookmark 提供代码中插入书签的功能,从而快速在页面间跳转。

  6. Code Issue Manager为团队提供issues和todo管理方便协同开发

  7. github copilot超强超快需要一些小钱10块用一年你也可以在github上申请student pack需要学信网认证和学生卡但有一定概率无法通过 在插件中你也可以找到一些免费的copilot替代品。推荐配合copilot labs一起使用其支持解释选中代码、修复选中代码中的bug、增加选中代码可读性、提高选中代码的稳定性等功能可以在你编写完代码后根据之前的代码记录和你的编写习惯高效地重构/优化/debug你的代码还可以一键生成文档和注释文档仅作参考你还是需要修改它自动以提高准确性

  8. ctrl+k ctrl+s配置属于你的快捷键,提高效率!

  9. live share和你的小伙伴一起结对编程



Ozone可视化调试和LOG功能

Ozone暂时只支持jlink。

22/11/16重要更新安装Ozone3.24 32-bit和J-Link7.22b目前可以支持Jlink和dap-link包括ATK无线调试器

软件安装

安装Ozone和J-link工具箱驱动、gdb以及各种调试工具。安装包都在网盘里。

注意如果希望支持daplink包括正点原子无线调试器请务必安装网盘对应的版本Ozone3.24 32-bit和J-Link7.22b)。

经过测试发现只有32位的ozone3.24支持daplink。

应该先安装Ozone再安装jlink。以下为步骤

  1. 安装Ozone

    image-20221116150122397

    这一步注意选择install a new instance安装一个新的实例。后续一路确认即可。

  2. 安装jlink

    image-20221116193340770

    这一步注意不要勾选update dll in other application否则jlink会把ozone里面老的驱动和启动项替代掉。choose destination和ozone一样选择install a new instance。如果安装了老的相同版本的jlink请先卸载版本相同不用管直接新装一个

  3. 替换动态链接库

    将网盘上下载的JLinkARM.dll放到JLink和Ozone的安装目录下替换原来的库。下载下来的库经过修改使得J-LinkOB在使用的时候不会报“The JLink is defective"和”you are using a clone version“的错误。

    之后如果安装其他版本的jlink也请注意==不要勾选==update DLL in other application否则会替换掉修改过的动态链接库。

配置调试项目

安装好两个软件之后打开ozone后会显示一个new project wizard如果没有打开在工具栏的File-> New -> New project wizard。

image-20221113133904084

选择M4内核为了能够查看外设寄存器的值还需要svd文件。所有mcu的svd都在图中的文件夹里提供当然你也可以使用我们仓库根目录下的文件。

image-20221116150901418

接口选择swd接口速度不需要太高如果调试的时候需要观察大量的变量并且使用日志功能可以调高这个值。如果连接了jlikn上面的窗口中会显示。如果链接了dap-link比如无线调试器会出现Unknown CMSIS-dap。选择你要使用的调试器然后继续。

image-20221113134252407

选择构建之后生成的.elf文件在项目文件夹下的build中。这是调试器专用的文件格式对其内容感兴趣可以自行搜索细节。此外ozone还支持.bin .hex .axf最后一个是amr-cc也就是keil的工具链会生成的等格式。

image-20221113134605331

这页不要动。如果希望保存jlink的调试日志最后一个选项选择一个文件或者新建一个日志文件。

启用FreeRTOS支持

注意如果你的代码使用了实时系统在载入项目的时候Ozone会进行对应的提示。选择载入支持实时系统的插件即可。

如果没有提示请在console中输入下面的命令然后回车即可

Project.SetOSPlugin(“plugin_name”)
# plugin_name是启用的实时系统支持插件名
# 我们要使用的命令是Project.SetOSPlugin ("FreeRTOSPlugin_CM4")

支持的插件在Ozone的安装目录下的Plugins/OS目录:

image-20221119174445067

我们的项目是F4的板子内核时Cortex-M4CM4因此选用FreeRTOSPlugin_CM4.js输入的时候js后缀不用输。 ozone默认输入的命令似乎有误需要手动修改这好像和ozone的版本有关请留意

常用调试窗口和功能

下图的配置是笔者常用的layout。每个窗口是否显示、放在什么位置等都是可以自己定义的。通过工具栏的view选项卡可以自行选择需要展示的窗口。

  1. 调试控制和vscode类似

  2. 变量watch窗口这里的变量不会实时更新只有在暂停或遇到断点的时候才会更新。若希望实时查看在这里右键选择需要动态查看的变量选择Graph他就会出现在窗口8的位置。

    如果不需要可视化查看变量变化的趋势但是想不暂停查看变量的值请右键点击变量选择一个合适的refresh rate

    image-20221119173731119

    如果是一个结构体,你可以为整个结构体都进行刷新率的配置,不需要手动一个个修改。或直接右键点击窗口将refresh打勾

    image-20221119173918340

  3. 断点和运行追踪管理

  4. 调试控制台,输出调试器的信息。

  5. 终端支持一些jlink script的命令。单片机通过log模块发送的日志也会显示在这里。

  6. 代码窗口用于添加断点、添加查看等。鼠标悬停在变量上可以快速查看变量值和类型。希望打开整个项目文件点击工具栏的view选项卡单击Source Files就可以打开一个项目中所有源文件的窗口。右键点击函数或变量可以跳转到定义和声明、查看汇编代码等。按F12跳转到定义。

  7. **变量可视化窗口这就是Ozone的大杀器。**在变量添加到查看watch之后右键点击watch中的变量选择Graph变量会被添加到可视化查看中。你可以选择“示波器”的显示时间步长以及颜色等信息还可以更改采样率。

    注意如果添加到动态调试窗口中没有反应请在窗口8中修改一下”Sample Freq“为100Hz或200Hz即可

  8. 窗口8和7配合。在窗口8中会实时显示变量值并且统计平均值和最大最小值而且还会将所有采样值保存到一个csv文件当中如果需要进一步分析可以导出这个数据文件。若要进行系统辨识、前馈设计等这无疑是最好的方法。这还可以方便我们观察采样值的异常进一步提升debug的效率。

  9. 内存视图。可以直接查看任意内存位置的值。

再次注意,这些窗口是否开启以及位置都是可以自定义的。

另外如果使用dap-link调试过程中可能会反复提示没有license请查阅附录1获取解决方案。

如果在调试过程中发现bug或者需要更改代码不需要终止调试或者关闭窗口。直接前往vscode修改并重新编译Ozone会自动检测到.elf文件的变化询问你是否重新加载项目。选择是后会自动开始下载并进入调试。

  • 变量动态查看(可视化)

    • 在变量的watch窗口右键点击变量选择一个refresh rate也可以实时查看变量和keil一样

    • 如果没有打开窗口现在view->timeline中打开可视化窗口。动态变量查看的窗口也在view->data sampling。

      启用动态变量查看的流程如下:

      graph LR
      在代码窗口中选中需要观察的变量 --> 添加到watch窗口 --> 在watch选择要动态查看的变量 --> 添加到Datasample窗口
      

      第一步的快捷键是ctrl+w,选中变量之后按。

      第二部的快捷键是ctrl+g选中watch中的变量后按。

      第三步可以修改示波器的步长和采样频率。

    • 如果当前文件没有你要的变量你想查看项目中的其他文件夹在view-> source files中可以打开该项目所有的源文件双击可以打开源文件。

      image-20221113142448939

  • 日志打印

在Terminal窗口查看还可以通过命令直接控制单片机的运行不过不常用

未打开窗口则在view-> terminal中打开。使用bsp_log打印的日志会输出到该窗口中.

  • 外设查看

在view-> register中打开窗口选择Peripherals可以查看所有外设寄存器

CPU选项卡可以查看CPU的寄存器。

  • 调用栈

在view-> call stack中打开窗口。

常用快捷键

组合 功能
ctrl+w 添加到查看
ctrl+g 添加到动态查看(需要先添加到查看)
f12 跳转到定义
f5 启动调试
f10 单步跳过
f11 单步进入
shift+f11 单步跳出
右键+break on change 当变量发生变化的时候进入此断点
ctrl+H 展示调用图,会列出该函数调用的所有函数(内部调用栈)

如果你使用拥有多个按键的鼠标,推荐将侧键前设置为ctrl+点击以查看声明/定义,侧键后设置为添加到watch(debug),侧滚轮设置为前进后退(历史)

你还可以按ctrl+K ctrl+S进入快捷键设置页面,将tab设置为下一个提示,用enter接受intelliSense建议,这样不需要将手移出主键盘区域. 将ctrl+;设置为移动到行尾,同时打开c/c++的函数括号不全,这样不需要手动敲击括号.

将alt+k设置为左移,alt+l设置为右移,这样不需要方向键.

选择最适合自己的配置!

保存调试项目

退出时可以将调试项目保存在项目的根目录下方便下次调试使用不需要重新设置。可以为jlink和daplink分别保存一套调试配置。

附录1为daplink添加license

在网盘上下载daplink_register_license.rar,解压出来之后打开。请关闭杀毒软件。

image-20221116152032104

根据Ozone打开时提示的daplink的序列号将其输入注册机电机generate就会生成5个license。

windows菜单搜索J-link license manager点击添加license将注册机生成的五个license依次复制黏贴并添加到的license manager中即可。

附录2在VSCode中启用SEGGER RTT日志

若使用Jlink进行调试只需要在开始调试之前把log任务启动便可以在终端中看到jlink rtt viewer的日志输出。

若使用daplink/cmsis-dap调试请使用Jlink调试任务启动,并且在调试任务启动之后再打开log任务

bsp_log.h中提供了不同的日志输出接口包括封装好的三种层级的日志info warning error和用户可以自定义输出格式的PrintLog()。还提供了一些简单的将浮点数据转化为字符串的函数方便进行日志输出。

附录3Makefile指令介绍

如果想要进一步学习Makefile可以参考这个链接Makefile Tutorial By Example。你会发现当项目越来越大的时候makefile也会变得复杂起来这就有了后继者CMake。cmake可以根据一定的规则生成makefile然后再利用make命令调用gcc进行程序的编译。也许以后还会有ccccmake

# makefile是CubeMX自动生成的,我们需要自己添加新编写的源文件路径和头文件文件夹,也可以额外加入自己需要的参数满足需求
######################################
# target
######################################
TARGET = basic_framework # 编译生成的目标文件名,如本项目会生成basic_framework.elf/bin/hex三个
# 注意,makefile会自动生成一个叫@的变量,其值等于TARGET.
# 在makefile中获取变量的值需要通过$(var_name),即加上括号并在前面使用$

######################################
# building variables
######################################
# debug build?
DEBUG = 1 # 是否启用debug编译.程序分为DEBUG版和RELEASE版,后者在编译时不会插入调试符号和调试信息相关支持的内容,使得程序运行速度提高.
# optimization
OPT = -Og # 编译优化等级,-Og表示调试级,常见的级别请看代码块下面的表格.



#######################################
# paths
#######################################
# Build path
BUILD_DIR = build # 编译的中间文件和目标文件存放路径,为了区分项目文件和编译输出,一般构建一个build(构建)文件夹,用于存放上述文件. 这个表达式也在生成了一个BUILD_DIR变量(可以把Makefile当作一种语言)

######################################
# source
######################################
# C sources, 参与编译的C源代码全部放置于此.注意如果换行写需要在行尾空格之后加反斜杠,最后一行不要加
# p.s. C语言的宏如果不能一行写完,也要在行尾加反斜杠,表示一行没有结束
C_SOURCES =  \
HAL_N_Middlewares/Src/main.c \
HAL_N_Middlewares/Src/gpio.c \
HAL_N_Middlewares/Src/adc.c \
HAL_N_Middlewares/Src/can.c 

# ASM sources 汇编源文件,第一个是stm32的启动文件,包含了bootloader的信息使得程序可以找到main函数的入口,第二个文件是添加对segger rtt viewer的支持.
ASM_SOURCES +=  \
startup_stm32f407xx.s \
HAL_N_Middlewares/Middlewares/Third_Party/SEGGER/RTT/SEGGER_RTT_ASM_ARMv7M.s

#######################################
# binaries, 下面是要执行的指令
#######################################
PREFIX = arm-none-eabi-  # 指令之前加的前缀,这里也是申明了一个变量
# The gcc compiler bin path can be either defined in make command via GCC_PATH variable (> make GCC_PATH=xxx)
# either it can be added to the PATH environment variable.
ifdef GCC_PATH # 和C语言的宏类似,如果在Makefile里定义或给make命令传递了GCC_PATH变量会执行以下内容.但实际上我们执行的是else的内容
CC = $(GCC_PATH)/$(PREFIX)gcc 
AS = $(GCC_PATH)/$(PREFIX)gcc -x assembler-with-cpp 
CP = $(GCC_PATH)/$(PREFIX)objcopy 
SZ = $(GCC_PATH)/$(PREFIX)size
else
# 定义了一个cc变量,其保存的内容实际上是gcc编译器的路径.makefile中要获取一个变量的值,需通过$(var).这里makefile会自动在环境变量里寻找gcc路径.CC里保存的内容是arm-none-eabi-gcc,就是我们添加到环境变量的arm gnu工具链的路径下的一个可执行文件.你可以尝试在cmd中输入arm-none-eabi-gcc,会发现这是一个可执行的程序.之前我们在验证安装的时候就运行了arm-none-eabi-gcc -v命令.
CC = $(PREFIX)gcc 
# 定义了一个AS变量,稍后会用于C/ASM混合编译
AS = $(PREFIX)gcc -x assembler-with-cpp 
# 定义变量.objcopy能够将目标文件进行格式转换.我们实际上要生成的目标文件是.elf,objcopy可以将其转化为hex和bin格式,用于其他用途.
CP = $(PREFIX)objcopy 
# size命令可以获取可执行文件的大小和包含内容信息.
SZ = $(PREFIX)size  
endif
HEX = $(CP) -O ihex # 这里用到了上面定义的CP,命令含义为将其转换成hex,i的前缀表示intel格式
BIN = $(CP) -O binary -S # 转化为二进制文件
 
#######################################
# CFLAGS, 在编译C语言程序的时候给GCC编译器传入的参数
#######################################
# cpu
CPU = -mcpu=cortex-m4 # 目标CPU类型.我们前面介绍过,不同的平台支持的汇编指令不同,一条相同的C语言表达式在翻译成汇编的时候会有不同的实现.比如8051单片机就只有加法器,因此他的乘除法都是通过多次加法和减法实现的,编译器就要完成这一工作.再比如STM32F4系列拥有浮点运算单元(FPU),可以直接在硬件上实现浮点数的加减法.这里指定编译的目标平台是cortex-m4内核的mcu.

# fpu 上面说了我们的f407是有FPU的,需要传入特殊的参数.fpv4-sp-d16表示float point,m4内核,single presicion, 16个dword(4字节)运算寄存器.
FPU = -mfpu=fpv4-sp-d16

# float-abi 使用软件还是硬件实现浮点运算.也就是我们说的如果没有FPU就只能使用软件实现浮点运算.这里选择hard硬件
FLOAT-ABI = -mfloat-abi=hard

# mcu 把上面几个变量合起来弄成一条长的参数
# Thumb是ARM体系结构中的一种16位指令集,这里-mthumb会启用它,感兴趣的同学可以进一步搜索.
MCU = $(CPU) -mthumb $(FPU) $(FLOAT-ABI)

# macros for gcc
# AS defines
AS_DEFS = # 汇编的一些宏定义

# C defines
C_DEFS =  \  # C语言的宏定义
-DUSE_HAL_DRIVER \ # 使用HAL库.HAL库的许多头文件和源文件里会判断是否定义了这个宏
-DSTM32F407xx \    # HAL库会根据使用的MCU的不同进行条件编译,这是一个很好的封装技术
-DARM_MATH_CM4     # 启用ARM MATH运算库,我们在卡尔曼滤波和最小二乘法的时候会用到矩阵运算

# AS includes
AS_INCLUDES = -IHAL_N_Middlewares/Inc
# 汇编包含目录.汇编语言也和C一样可以多个文件联合编译,在没有C语言的时候大家都是利用这种方式开发的.在一些运算资源极其受限的情况下也会直接编写汇编.
# CubeMX生成的HAL包含目录Inc下有一些头文件里面就包含了一些汇编要用到的头文件

# C includes, C语言的包含目录,将所有参与编译的头文件目录放在这里,注意是目录不需要精确到每一个文件.
# 不想一行写完记得行尾加\,最后一行不要加
C_INCLUDES =  \
-IHAL_N_Middlewares/Inc \
-IHAL_N_Middlewares/Drivers/STM32F4xx_HAL_Driver/Inc


# compile gcc flags, gcc的编译参数,这些参数自己感兴趣的话去搜索一下.这还将之前定义的一些参数以变量的形式放过来.
ASFLAGS = $(MCU) $(AS_DEFS) $(AS_INCLUDES) $(OPT) -Wall -fdata-sections -ffunction-sections

CFLAGS += $(MCU) $(C_DEFS) $(C_INCLUDES) $(OPT) -Wall -fdata-sections -ffunction-sections

ifeq ($(DEBUG), 1)
CFLAGS += -g -gdwarf-2
endif


# Generate dependency information
CFLAGS += -MMD -MP -MF"$(@:%.o=%.d)"


#######################################
# LDFLAGS,传递给链接器的参数
#######################################
# link script
LDSCRIPT = STM32F407IGHx_FLASH.ld # 需要参与链接的文件.这个文件指明了特定MCU的内存分布情况,使得链接器可以按照此规则进行链接和地址重映射.

# libraries,要添加的库,这里我们要使用编译好的math运算库.在CubeMX里面生成的时候可以在第三方库选择DSP运算库,生成makefile时会自动添加进来.
LIBS = -lc -lm -lnosys  \
-larm_cortexM4lf_math
LIBDIR =  \ # 和上一行命令对应,这里引入库的目录,gcc会自动去目录里寻找需要的库文件
-LHAL_N_Middlewares/Drivers/CMSIS/Lib/GCC
LDFLAGS = $(MCU) -specs=nano.specs -T$(LDSCRIPT) $(LIBDIR) $(LIBS) -Wl,-Map=$(BUILD_DIR)/$(TARGET).map,--cref -Wl,--gc-sections

# default action: build all
all: $(BUILD_DIR)/$(TARGET).elf $(BUILD_DIR)/$(TARGET).hex $(BUILD_DIR)/$(TARGET).bin


#######################################
# build the application
#######################################
# list of objects
# OBJECTS保存了所有.c文件的文件名(不包含后缀),可以理解为一个文件名列表.notdir会判断是否是文件夹
OBJECTS = $(addprefix $(BUILD_DIR)/,$(notdir $(C_SOURCES:.c=.o)))
vpath %.c $(sort $(dir $(C_SOURCES))) # 对.c文件进行排序,百分号%是通配符,意为所有.c文件vpath是makefile会搜索的文件的路径.如果最终找不到编译中产生的依赖文件所在的路径且不指定搜索路径makefile会报错没有规则制定目标(no rule to build target)

# list of ASM program objects
# 把所有.s文件的文件名加到OBJECTS里面
OBJECTS += $(addprefix $(BUILD_DIR)/,$(notdir $(ASM_SOURCES:.s=.o)))
vpath %.s $(sort $(dir $(ASM_SOURCES))) # 对.s文件的文件名也进行排序

# 以下是编译命令,命令之前被高亮的@就是静默输出的指令.删除前面的@会将输出显示到命令行.
# 如@$(CC) -c $(CFLAGS) ...... 去掉第一个@即可.

# 意味根据makefile,在BUILD_DIR变量指定的路径下将参与编译的所有.c文件编译成.o文件
$(BUILD_DIR)/%.o: %.c Makefile | $(BUILD_DIR) 
 @$(CC) -c $(CFLAGS) -Wa,-a,-ad,-alms=$(BUILD_DIR)/$(notdir $(<:.c=.lst)) $< -o $@
 # 上面这句话翻译一下实际上是gcc -c -many_param build/xxx -o build
 # 意思是将所有参与编译的文件都列出来,传递一堆编译参数,让他们生成.o文件,并且放在build文件夹下
# 意为根据makefile,将.s文件编译成.o文件,具体和上一条命令差不多
$(BUILD_DIR)/%.o: %.s Makefile | $(BUILD_DIR)
 @$(AS) -c $(CFLAGS) $< -o $@
# 根据前两步生成的目标文件(.o,这些文件的名字保存在OBJECTS变量里),进行链接生成最终的.elf
$(BUILD_DIR)/$(TARGET).elf: $(OBJECTS) Makefile
 @$(CC) $(OBJECTS) $(LDFLAGS) -o $@
 @$(SZ) $@ # 输出生成的.elf文件的大小和格式信息

$(BUILD_DIR)/%.hex: $(BUILD_DIR)/%.elf | $(BUILD_DIR)
 $(HEX) $< $@ # elf转换成hex
 
$(BUILD_DIR)/%.bin: $(BUILD_DIR)/%.elf | $(BUILD_DIR)
 $(BIN) $< $@ # 转换成bin
 
$(BUILD_DIR): # 如果makefile所处的文件目录下没有build文件夹,这里会新建一个build文件夹.
 @mkdir $@ 

#######################################
# clean up,清除编译信息,可以在命令行中通过rm -r build执行,实际上就是把build文件夹删掉
#######################################
clean:
 rm -r $(BUILD_DIR)
# 你的makefile可能会使用cmd而不是powershell来调用内核,而cmd不支持rm命令,因此可能要修改为rd (remove directory),cmd传入参数的方式为 /x  ,x为要传入的参数

  
#######################################
# dependencies
#################################
######
-include $(wildcard $(BUILD_DIR)/*.d) # 包含所有的依赖文件(d=dependency),这是编译产生的中间文件,当hello.c包含hello.h而后者又包含了其他头文件时,会产生一个hello.d,它包含了hello.h中包括的其他的头文件的信息,提供给hello.c使用.

# *** EOF ***

另外我们还在仓库的根目录提供了Makefile.upgrade文件在该makefile中我们提供了一些基于命令行的更高级的特性例如自动源文件索引和头文件目录包含等使用之后若添加了新的源文件就不再需要手动添加路径了。这里面还增加了一些为嵌入式选择的gcc优化参数。要启用高级特性将其内容复制到你的makefile即可。

  • 编译优化等级:
优化级别 说明 备注
-O0 关闭所有优化 代码空间大,执行效率低
-O1 基本优化等级 编译器在不花费太多编译时间基础上,试图生成更快、更小的代码
-O2 O1的升级版推荐的优化级别 编译器试图提高代码性能,而不会增大体积和占用太多编译时间
-O3 最危险的优化等级 会延长代码编译时间,生成更大体积、更耗内存的二进制文件,大大增加编译失败的几率和不可预知的程序行为,得不偿失
-Og O1基础上去掉了那些影响调试的优化 如果最终是为了调试程序,可以使用这个参数。不过光有这个参数也是不行的,这个参数只是告诉编译器,编译后的代码不要影响调试,但调试信息的生成还是靠 -g 参数的
-Os O2基础上进一步优化代码尺寸 去掉了那些会导致最终可执行程序增大的优化,如果想要更小的可执行程序,可选择这个参数。
-Ofast 优化到破坏标准合规性的点(等效于-O3 -ffast-math ) 是在 -O3 的基础上,添加了一些非常规优化,这些优化是通过打破一些国际标准(比如一些数学函数的实现标准)来实现的,所以一般不推荐使用该参数。

附录4VSCode直接烧录代码

有时候你对自己的代码特别自信不想debug想直接下载代码那么直接通过openocd或J-Flash即可随jlink一起安装。要是觉得这样有点麻烦还要再开一个软件他们两者都支持通过命令行执行。你可以在vscode的tasks.json中编写一个额外的任务来实现。这里我通过给Makefile添加伪构建目标来利用make命令执行下载操作

#######################################
# download without debugging
#######################################
OPENOCD_FLASH_START = 0x08000000 # 如果切换芯片可能需要修改此值

download_dap:
 openocd -f openocd_dap.cfg -c init -c halt -c "flash write_image erase $(BUILD_DIR)/$(TARGET).hex $(OPENOCD_FLASH_START)" -c reset -c shutdown

download_jlink:
 openocd -f openocd_jlink.cfg -c init -c halt -c "flash write_image erase $(BUILD_DIR)/$(TARGET).hex $(OPENOCD_FLASH_START)" -c reset -c shutdown

首先设定了flash烧录区的起始地址下面两个构建目标分别用于daplink和jlink的下载。我们统一使用openocd进行烧录。命令中-c表明的是启动openocd之后要执行的命令openocd作为一个gdbserver是用作调试的因此这里我们在flash write_image之后直接reset让单片机复位开始运行程序,然后立刻退出调试,从而达到下载程序运行但不调试的目的。

若你认为在makefile中使用伪构建任务不合适也可以自行在task.json中编写一个任务。

接下来我们希望能够直接下载,不要在命令行里面输入make download_dap这么复杂的指令我们可以利用make构建伪造目标来实现命令行命令执行因此在tasks.json中添加如下两个任务

 {
     "label": "download dap",
     "type": "shell",
     "command":"make download_dap",
     "group": {
      "kind": "build",
      "isDefault": false,
      },
 },
 {
     "label": "download jlink",
     "type": "shell",
     "command":"make download_jlink",
     "group": {
         "kind": "build",
         "isDefault": false,
         }
 },
// 实际上也可以直接编写命令行指令,他们是等效的

这样在工具栏的Terminal页面就可以选择对应的任务直接下载执行了。你也可以通过快捷键ctrl+shift+B唤起任务执行页面进行选择。如果你希望立刻检验你代码修改的效果,在下载之前进行编译,那么在command信息下新增加一个mingw32-make -j24即可,或者添加一个preLaunchTask。对于调试如果不想点两下想在修改代码之后直接调试也可以在launch.json中增加preLaunchTask(文件中已经添加,需要的话取消注释即可)。

实际上换用arm gnu工具链之后可以指定多线程编译因此消耗的时间非常短建议都加上先编译的选项不会占用额外的时间。可以把默认task设置成编译后下载把默认debug设置成编译后调试提升开发效率。

附录5利用MSYS2安装依赖环境

之所以要使用Linux进行C++开发是因为在开发环境中配置依赖包、依赖应用和库非常的方便。Debian系有aptFedora和Redhat系有yum他们都可以方便地帮助我们下载开发软件必须的一些文件和工具。在windows的宇宙最强IDEVisual Studio中配置头文件和动态链接库可以称得上是最折磨的事。好在现在Windows下也有可以使用的包管理工具了MSYS2

安装包已经上传到了网盘的EC/VSCode+Ozone环境配置/msys2-x86_64-20221028.exe你也可以在msys2的官网直接下载如果没有ladder速度可能会稍慢镜像站的下载速度会快许多。安装之后打开MSYS2 MSYS软件他是一个类shell的界面实际上它提供了包括mingw64、ucrt64、clang64等多种不同编译环境在内的一组类linux开发工具合集

image-20221119222946103

输入以下命令然后一路回车即可:

pacman -S mingw-w64-x86_64-toolchain mingw-w64-x86_64-arm-none-eabi-toolchain mingw-w64-x86_64-ccache  mingw-w64-x86_64-openocd
# 需要注意ctrl+V不是黏贴快捷键,而是Ins+Shift.或者右键点击空白处选择黏贴也可以.

image-20221119223148604

比如上面这样,会让你选择,直接敲回车即可,等待安装

刚进来第一次安装可能还会更新一下数据库,也是全部更新就行。

安装好之后把Msys2下的mingw64的bin加入PATH环境变量D:\Msys2\mingw64\bin(这是我的路径,注意要选自己的)。

注意,如果选用此安装方式,OpenOCD的可执行文件也会被放在上面这个路径下记得稍后在VSCode中配置的时候找到这里。相应的scripts放在D:\Msys2\mingw64\share\openocd下。

通过这种方式安装之后还可以选用ccache加速编译。ccache会根据之前的编译输出建立缓存使得之后编译时可以直接读取缓存。要开启这个功能直接在Makefile中搜索PREFIX将下面一行的内容替代原有内容即增加ccache在arm-none-eabi-之前)。

  • 将msys2的各种终端集成到vscode当中

当你的系统中存在各种不同的编译环境时将他们的路径一同添加到系统环境变量PATH中可能导致错误的工具和指令调用也有可能在编译中匹配错误的库和头文件。为了防止这种事情发生最好是只在特定的工具集中添加它们需要的环境变量。Msys2提供了各种子环境的终端shell你可以将它们集成到Vscode的终端中当要进行对应的开发时就新建一个你需要的环境的终端。

ctrl+,打开vscode设置或直接打开vscode设置的.json文件添加以下内容便会将对应的环境终端集成在vscode中

  "terminal.integrated.defaultProfile.windows": "mysys2-mingw64", // 若希望为默认终端可以加入这一句
  "terminal.integrated.profiles.windows": {
    "mysys2-mingw64": {
      "path": "cmd.exe", // 意思是使用cmd作为父进程启动终端
      "args": ["/c","C:\\msys64\\msys2_shell.cmd -defterm -mingw64 -no-start -here"]
        // msys2_shell.cmd在msys的安装目录下这里要改为你自己的目录
    }
  }

修改好之后便可以在vscode中使用了

image-20230723132711865

linux下熟悉的ls/mkdir/find/ld/cat等命令和工具也一应俱全了。

不过这时配置的build task和其他download task仍然在powershell或cmd中启动若没有添加环境变量它们无法找到对应的可执行文件这时候你就要修改task.json将使用的shell改为mingw64即可若你使用ucrt或clang同理

附录6Windows修改用户名为英文

  1. 右键【任务栏win徽标】->【计算机管理】->【本地用户和组】->【用户】->右键【中文用户名】->【重命名】

    如果是win10则是右键【计算机一般放在桌面上如果没有的话桌面右键选择个性化然后打开桌面图标】->【管理】->【本地用户和组】->【用户】->右键【中文用户名】->【重命名】

    注意这个英文用户名后续还要使用

  2. 以管理员权限启动命令提示符,输入

    net user administrator /active:yes
    

    如果杀毒软件提示恶意修改,请放行

  3. 重启电脑,登录的时候选择Administrator账户,需要进行一些简单的设置,直接一路确定即可。

  4. 进入C盘的用户文件夹下把你原来的用户文件夹名称改成刚刚设定的英文用户名。

  5. 【Windows+R】->【运行】->输入 regedit ->回车

    找到HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Profilelist,把这个标签页下每个选项都点开看,其中有一个页面对应的ProfileImagePath值是以你用户名命名的文件夹路径,比如我的路径是C:\User\小曾

    双击,把这个路径改成你新设的用户名,比如我的C:\User\Neo

  6. 重启电脑,进入修改好的用户,

  7. 以管理员权限启动命令提示符,输入

    net user administrator /active:no
    

    如果杀毒软件提示恶意修改请放行。这会关闭Administator账户。