你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

C++ 指南也可以在STM32 上使用!

[复制链接]
eefishing 发布时间:2019-12-17 12:47

简介

本文描述了如何使用在搭载了 RT-Thread 系统的 STM32 平台上使用 C++,包括 C++ 的配置和应用等。并给出了在STM32F411 NUCLEO开发板上验证的代码示例。

4 i2 H( x9 |3 h/ I6 i) j9 n: N1 i硬件平台简介
% `; v( U& D0 F8 E+ F+ S  K3 l
: q/ c( D, T/ k
本文基于意法半导体 STM32F411 NUCLEO开发板,给出了 C++ 的具体应用示例代码,由于RT-Thread上层应用API的通用性,因此这些代码不局限于具体的硬件平台,用户可以轻松将它移植到其它平台上。: X% a* P1 m! C0 G
STM32F411 NUCLEO是意法半导体推出的一款基于ARM Cortex-M4内核的开发板,最高主频为100Mhz,该开发板具有丰富的板载资源,可以充分发挥STM32F411RE 的芯片性能。
STM32F411RE从属于销量名列前茅的STM32F4系列,众所周知,F4是STM32主打高性能和数字信号处理的“轻奢”系列。
“奢侈”在F4作为内核为Cortex-M4 (DSP+FPU)的MCU,可选180MHz 主频、2M Flash/384KB RAM、Chrom-ART加速器、MPI-DSI接口、延伸到125度的工作温度、DFSDM数字滤波器以及各种常见的音频(SAI)、连接(Ethernet、Camera、USB)、控制(CAN、UART、I2C)、存储(FMC、2/4/8 bits SPI、SDMMC)外设。
“轻”在价格让人“轻松”、尺寸“轻巧”(不到3mm*3mm的封装)、功耗“轻微”。

1.png

如何在 STM32 上使用 C++
7 I/ h. t; x4 y% L* A
准备工作

4 Q6 f) s# k4 z' u# b* G

1、下载 RT-Thread 源码

2、下载 ENV 工具

3、进入rt-thread\bsp\stm32f411-st-nucleo 目录,检查 BSP rtconfig.py 文件和 SConstruct 文件是否支持 C++ 配置,如下图所示

检查 rtconfig.py 文件中对 C++ 的支持

2.png

检查 SConstruct 文件中对 C++ 的支持

3.png

打开 C++ 支持:

打开 Env 工具,在 Env 命令行中输入 menuconfig,进入配置界面,使用 menuconfig 工具(学习如何使用)配置工程。在 menuconfig 配置界面依次选择 RT-Thread Components ---> C++ features ---> Support C++ features,如图所示:

. n; H% F  f5 {+ q

4.png

编译工程: scons --target=mdk5 1. 生成 mdk5 工程,将示例代码附带的 main.cpp 替换掉 BSP 中的 main.c 并重新加入到工程中,如图所示:

5.png

编译,下载程序,在终端输入 help 命令可以看到 test_cpp 已经添加成功了。

9 g9 K# M! D3 A2 V/ G

6.png

运行 C++ 程序:

在终端输入 test_cpp 运行结果如下图所示。

7.png

6 R4 I0 {, `: T* o7 s; C

. R* u6 R' B5 C
+ p, A$ h- q. T$ Y7 E+ C' cC++ 全局对象构造函数的调用: w% @" i5 u3 {$ ?! m

0 q4 r( u, U7 Q/ ]& t$ e2 F- S, m9 y/ A* N8 K: T% M: V3 S

RT-Thread 中对全局对象构造函数的实现中,以 GNUC 为例,在 rt-thread\components\cplusplus 目录下的 crt_init.c 文件中对 C++ 进行了系统初始化, 在特定的 BSP 目录下,连接脚本文件 link.lds 为 C++ 全局构造函数的代码分配了段,使 C++ 全局对象构造函数链接后能够存放在指定的段中。如下图所示:

8.png

crt_init.c 文件完成了 C++ 系统的初始化工作

C++ 系统初始化部分:

* o- h5 R1 ^5 u8 Y. C/ F/ L6 ~+ s% ~
1RT_WEAK int cplusplus_system_init(void)) h0 k' f- @3 j, B; J
2{
" c3 D; r& e' _0 C5 T0 ? 3    typedef void(*pfunc)();: T% l' i$ K+ @$ m! k
4    extern pfunc __ctors_start__[];( \: }+ f) X9 w% q1 n/ U4 z
5    extern pfunc __ctors_end__[];
2 z, |% p/ p, Z, J. t 6    pfunc *p;( K/ W# g$ g( A7 x$ p) i
7* p* V4 \5 J  Y7 a
8    for (p = __ctors_start__; p < __ctors_end__; p++)
9 ~# M0 {" e; t  \+ x 9        (*p)();
0 w1 w0 x2 x. a* j) N% O; F10
3 P8 L1 R  Q6 S: {0 C9 {11    return 0;
* d: d& o. h. H& S12}( G( F0 E! R3 n' P! |
13INIT_COMPONENT_EXPORT(cplusplus_system_init);
/ W! T0 L- s/ @: q$ q; g/ D

在 cplusplus_system_init 函数中,将全局对象的构造函数依次链接到了链接脚本文件中为其分配的段中,并且调用了 RT-Thread 组件自动初始化的宏 INIT_COMPONENT_EXPORT,所以在链接的时候,C++全局对象构造函数所产生的目标文件就被链接到了__ctors_start__和__ctors_end__组成的段中。
0 R3 }% C% C* O1 X

链接脚本中为 C++ 全局构造函数分配的段部分:

! \% C. ~4 H5 v. F9 X* Q
3 s7 e6 E$ [! e" o" H8 l" z1PROVIDE(__ctors_start__ = .);
3 M5 u" o. @7 b+ w2KEEP (*(SORT(.init_array.*)))
. t8 h0 \* @5 u& Y3KEEP (*(.init_array))  m8 U4 W( F3 n1 k0 @1 R5 J
4PROVIDE(__ctors_end__ = .);
, g. O) c% l% G' v# e- e  Q: _$ ?# b* F; m2 |

__ctors_start__ 分配了 C++ 全局构造函数段的起始地址, __ctors_end__ 分配了 C++ 全局构造函数段的结束地址,所以全局构造函数在系统初始化的时候,就会被链接到这里分配的段地址中。

RT-Thread C++ 异常说明
$ g6 s' x6 ^' i4 J7 m+ R

同样,在链接脚本文件 link.lds 中,也为 C++ 异常分配了段地址:


5 u$ J8 L3 x7 @' X1    __exidx_start = .;% g# G5 t& Q1 C) p' o8 I" l+ p
2    ARM.exidx :4 I% a3 f! V. f: h. h6 _. i4 R$ }! X
3    {- p/ T  N# z/ O7 N$ O+ i
4        *(.ARM.exidx* .gnu.linkonce.armexidx.*)
; e" V& H, S# S; `) d* [2 ?. _5        _sidata = .;
5 @: c7 X) U( K" {: T6    } > CODE4 z3 I' V+ W: ?+ [1 ^3 Q, t/ |) a& k
7    __exidx_end = .;2 \% G9 O6 J! T" P2 C

" Z$ {* b- s: t5 P- e9 o9 x3 a4 E' W

__exidx_start 分配了 C++ 异常的起始地址, __exidx_end 分配了 C++ 异常的结束地址,当异常产生的时候,就会被分配到指定的段地址中。

这里以一个 C++ 除零异常的抛出和捕获为例:


/ z, D4 J4 r6 A0 N/ f: v 1   #include<math.h>/ ^8 P: H8 b- W. V/ ~. U/ A
2
( q1 `* D, X' C 3    #define MIN_VALUE                 (1e-4)                          4 Q- a9 W, R+ Z! M2 }
4    #define IS_DOUBLE_ZERO(d)         (abs(d) < MIN_VALUE)9 ~# k/ Z: J- R  x0 l* C2 t
5
% p& V3 F6 U1 g/ Q/ |7 [8 l# C 6    double div_func(double x, double y)                     3 r( M) M6 ~+ n5 E* E+ I& `- b
7    {- T9 d6 q/ m$ R) J7 t% @; g1 c
8        if (IS_DOUBLE_ZERO(y))' \* \. U, A+ |; j  q& W5 Q
9        {9 }2 s6 s5 X0 I& _- Y
10            throw y;                                           /* throw exception */
3 P. t3 u1 E  g4 A0 \& C9 R11        }9 @! T8 h0 E. D2 f* ?; }6 g
12
) z4 {( i3 p  B1 B( Z3 G13        return x / y;                                 
) u! C, p7 ]$ r% Q8 r/ I- V& w14    }4 x) [2 k- U" I- Z! X; i
150 U0 e" R2 h4 F% q7 j% W
16    void throw_exceptions(void *args)# A9 {  p1 a/ ~2 l: e/ U! Z! [! ^
17    {
9 d. j3 B) K# G18        try                                            
1 S( r% E) D: P19        {  f! H, ~" S6 ^+ `0 h/ K# V* w- _
20            div_func(6, 3);9 @* @$ \3 Z& ?. \& y, j! N; ?
21            rt_kprintf("there is no err\n");
7 P" L1 T! f" q+ S4 U7 x6 x* O22            div_func(4, 0);                                   /* create exception*/
6 {4 i. J7 `: D6 T23            rt_kprintf("you can run here?\n");0 Z; t5 q% d0 ~7 \  _% R
24        }
( \/ A# A# n/ s9 |* }25        catch(double)                                         /* catch exception */     
% O+ n" l5 o. g& f26        {
0 b. X; z& x/ n3 o+ p# v" H! ]27            rt_kprintf("error of dividing zero\n");
( @! W  K* t) R  d- g: t28        }
5 F1 F3 z+ \/ c0 h) W  G! s; p0 k29    }
1 e) _; n3 ^6 V" C4 W. ~  C302 ?) d( r2 W9 e) e
31    MSH_CMD_EXPORT(throw_exceptions, throw cpp exceptions);
- M! i6 A! Q+ g0 \5 x3 P! Q. V5 K$ N4 N7 \9 W: D$ }) g. z  H

  c; W# z" q& L7 |6 W0 Y

当除零异常发生的时候 div_func 函数会抛出一个异常,在 throw_exceptions 函数中会去捕获这个异常。

下载代码,并在终端输入 throw_exceptions 运行结果如下图所示。

9.png

到这一步为止,如何在搭载了 RT-Thread 系统的 STM32 平台上如何使用 C++ 的介绍就结束了。

( G' }2 M# s$ z5 O- |
参考资料) K+ M$ R0 ?2 @& D6 E3 M
9 e3 _, r. `# n4 i; Q

1、ENV 用户手册! z+ ^$ T" U$ A! F8 m* b/ o$ E+ _6 n

http://www.rt-thread.org/document/site/programming-manual/env/env/

3 }% w8 G! U9 _5 r. q8 s

2、STM32F411-ST-NUCLEO BSP 源码

http://github.com/RT-Thread/rt-thread/tree/master/bsp/stm32/stm32f411-st-nucleo

(请将以上链接复制至外部浏览器打开)

( O: v) U2 @- h; |
收藏 1 评论2 发布时间:2019-12-17 12:47

举报

2个回答
w014017 回答时间:2019-12-17 14:13:24
pkoko 回答时间:2019-12-18 12:06:30
学习了,感觉使用C++比C方便些。不过目前的资源还比较少

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版