本文是STM32U0测评系列的最后一篇。我认为STM32U0最吸引我的就是他极致的低功耗,ST官方也说他非常很适用于水表等产品的应用,于是我只做了一个电子墨水屏RTC日历。
" e+ r3 z0 q. a. U. W
( c5 g) ~" Q; n2 R1 J: ?: q- C z4 q$ O5 O
一开始我是想做日历+时钟+温湿度。但是很可惜我手上的这块电子墨水屏不能局部刷新,只能全刷,而全刷要15s,也就是说1分钟内只有45s是正常显示的,这也太鸡肋了,完全没有实用性可言,所以最终决定就只做日历,一天刷新一次,功耗也是嘎嘎低
" I; F9 B" D+ k$ Z$ c/ y+ t3 E8 L0 L" }/ t
/ K2 d8 J1 o+ N5 `由于中途有很多调试工作,我就不再一步一步讲解是怎么做的,简单讲解一下demo功能、接线、代码逻辑、上位机使用注意、后续改进想法等0 P7 T% b! J- p2 F& |
* }7 J( d* [3 C0 H$ f: Z( u
8 M- E( G4 I. ^' f一、简述功能
/ W' ^% ^# ?3 q使用STM32U08 NUCLEO开发板外接一块我自己做的电子墨水屏驱动板(软硬件已开源,地址:https://github.com/BUYITAO/My_E-Paper_Driver),显示当前的日期,每天24点整由RTC闹钟唤醒刷新屏幕,平时MCU都处于STANDBY,这样可以让整个系统处于最低功耗状态。正常显示如下(偷懒了,没有做界面美化设计,请见谅 )
5 F0 i, {6 G6 a9 R, G% d
' a2 a0 a8 P% l. k
9 D U4 _# t* T8 F" A3 u. o
系统首次上电后由于RTC参数丢失,会处于默认的预设状态,需要用户手动设置一下时间,此时电子墨水屏显示“Please config RTC data”,如下图所示
( |$ h* y6 e, g) G9 D7 ]( Q
. Y' ]5 Q) ]/ p% w: J: \
/ ?, \# A! z' U& J% x% R* K3 O. I: h+ ?+ y I
用户可以通过串口发送配置参数(这个我做了一个上位机可以一键发送配置指令),MCU收到后,刷新屏幕,再进入standby。上位机界面如下图
: }! I6 A1 g8 e8 C $ F8 z; B' z/ L
, e( a' x( J2 t- A
, D) _. _* G4 F) s }) f3 I
3 C% b* j1 }8 a+ d* @4 P* k
7 W- E0 h, c) B9 X8 S- i1 P二、硬件连线 8 L& C2 I+ ?2 {7 y8 p, {
; A1 E* g9 ]( }7 n) L+ Z( ]: @
& ^, S/ h, p6 R3 _. j0 d0 {/ W
" A' a# a& e7 o: L& K4 u墨水屏驱动板 STM32
, B* o$ {5 K* s- D1 JMOSI PA7 $ U. ^4 Q2 r" t) c" [$ e
CS PC9 2 P* S! _9 B1 I* d* a8 M l/ g
RST PC8
1 y2 p+ y) B+ s1 T2 X6 VCLK PA1
) D8 l7 q1 O% s: g' e3 S$ w& MDC PC6 # J8 I3 ~" O8 u
BUSY PC5 * G5 S6 H9 R$ G K1 H
/ J0 g- }- `% e, H
% e- u) e" T" r$ n9 x% W
照片不太看得清楚开发板上的接线,我把原理图放上来 2 D# U7 ?: x. b: a
# a* l6 N; l0 t: i! D: v1 Z
4 B# w6 }2 F- |0 x: Y7 x" h& V( V: t
1 h1 B/ \+ U4 ?( \' u+ L6 b
然后串口部分我借用了STlink的VCP(这个最后改进点部分我会讲到,用这个其实对功耗影响挺大的,后续打算改掉) ! l' Y9 a: G+ p4 f" [" q4 C, T
9 E% g" i1 P; X ^: _1 z ^
; ~9 E4 W6 t7 }, B& ~' N
三、代码简单讲解
7 ?2 \" }2 B% _ ?/ A" h; Z
. V8 s+ |6 R' \CUBEMX打开SPI,然后配置了GPIO。这样就可以实现与电子墨水屏的通讯
" ^5 T2 \& G r# p# u, n$ Q
5 |- a9 M* p; V- @: H+ z8 d1 r
% o7 }$ i6 M3 K" x: P( a+ O. [5 L
; C) e J/ X) r
9 x& ~% N' q8 f; l. V. d
\: A# g* x, O0 N+ N/ U+ s
3 q( Z8 q6 t5 S- `; l9 Z; M3 b. u然后把GPIO和SPI的库改成LL库(个人习惯,这些基础的外设以前LL库用惯了,毕竟可以看代码学习对应寄存器的操作),然后生成代码即可 ) h& i8 Q! \! V9 o. o( C- D
- X5 P# o; ?+ E( B8 B
, U7 ~- `: G( T2 c5 A, x***这里需要注意一下生成的代码有两个BUG,毕竟现在U0的PACK才第一版,有点BUG很正常,希望ST可以看到我的这个文章,在后续的版本中修复一下
/ \, w) ?* r' m3 A: v \0 M第一个是在main.h中,红框这个没有换行,会导致编译报错 $ `- t: H3 ]5 N8 V4 Y# i2 |
# F7 j' n, y, A! ^" O, A* N8 C, n# @
5 I3 o4 }4 S0 F
第二个是生成的工程会强制设置为V6编译器,V6编译会疯狂报错,改成V5就好 " I9 P9 J( v8 ^! ?" Q
: _) c8 |% v" \4 N" N ]! c: A
6 ^0 l% f$ Y6 } I9 B1 z3 l2 x- A3 Q6 c; Q7 p, W
% ^0 h2 i, u E3 _
# [; c) _7 e+ Q" }) E4 e% R, H然后关于电子墨水屏的移植、配置啥的我就不细讲了,在我分享的github上有详细的readme。接下来简单看一下功能代码
0 U; f. R( k8 d$ l- Hwhile1前我会初始化墨水屏,然后判断是否为首次上电。如果是首次上电,就让墨水屏显示“Please config RTC data”,反之就清除标志位,刷新屏幕(到这里就是RTC闹钟到时间了,要刷新屏幕,显示新的日期),然后再进入Standby - /* 初始化墨水屏 */' W* g7 k" u$ D! @
- E2213JS0C1_Init(0);
# w" T z }) T0 W6 c - /* 是否为首次上电 */
7 e2 A( v z# S9 y/ e4 O6 @" ?& _ - if(__HAL_PWR_GET_FLAG(PWR_FLAG_SB) == RESET). q, [3 ?# h, E1 O2 F, J! e
- {
9 k6 v$ e/ X7 N- } - printf("normal run\r\n");8 [: z. J U. }2 ]3 D% |9 Q& [* B
- /* 打印RTC时间 */
* r+ Q- i8 A: D" c* U. a- K( G' R - print_rtc_data();
* _9 C) P+ N) E5 f* y! }3 Y - E_Paper_show_first_power_on_page();
* i2 _) X8 T# \$ Z/ G+ l - }
. Y/ x% n: k1 I t: G - /* 从standby唤醒后的复位 */6 i: Y1 c6 E# i, w$ H: H; J1 N$ V
- else
" q# H) `$ @/ m# O - {5 W0 `, s' b; d0 N, a
- printf("wkup from standby run\r\n");
- k! a7 N5 k. V+ w1 } - /* 清除standby的标志位 */( l% X9 S f6 W; e/ }: S# a
- __HAL_PWR_CLEAR_FLAG(PWR_FLAG_SB);/ w4 f. X: C$ P/ n, A
- /* 清除闹钟标志位 */
/ E& K7 P. o- N* \& H" y5 ]% | - __HAL_RTC_ALARM_CLEAR_FLAG(&hrtc, RTC_FLAG_ALRAF);
3 ?; Z2 `6 r I" G5 L. b - /* 刷新屏幕 */
! I `! V, ^% D/ z - E_Paper_show_calendar();7 H9 q. f) G: Z8 C2 k" L/ w* I
- /* 进入standby模式 */6 z' N* ` r V2 ~3 T2 ]
- HAL_PWR_EnterSTANDBYMode();- {$ N5 j/ v. d, ~ d- p
- }
复制代码
% A# {# O {/ @4 @# g k
6 f9 n/ q$ b& Z3 e5 [! @1 D( p' z) V
如果是首次上电的,可以看到没有调用standby函数,所以他可以向下进入while1,while1中的函数如下 - if(HAL_UART_GetState(&hcom_uart[COM1]) == HAL_UART_STATE_READY)5 m k# I$ r6 m+ e, n
- {
. J# }* t. l# u$ P/ g" } - if(HAL_UART_Receive(&hcom_uart[COM1], (uint8_t*)&buffer[idx], 1, HAL_MAX_DELAY) == HAL_OK)
. o# ~4 B8 r1 t - {
# X/ N# C& c: a2 ]8 Y - if(buffer[idx] == '\n') // 假设\r\n'作为结束标志& k" B8 h O$ y a, G6 V
- {
' A. o( u- b9 ]/ V% |/ t9 L - buffer[idx] = 0; // 添加字符串结束符% P8 v8 F( l' U4 O6 }& E3 J
- idx = 0; // 重置缓冲区索引
5 L9 {3 T I5 S3 R' R+ y - if(ParseATCommandAndSetRTC(buffer))
& _6 v/ S: w/ R; S& r8 R - {! ^- _1 W1 Y2 z
- printf("Parse AT Command success\r\n");4 G5 M* t1 v# N4 |
- /* 刷新屏幕 */
/ l9 P5 y2 H/ h8 ? - E_Paper_show_calendar();
1 Z" f1 M! b% c" W4 y+ Q' J. b - printf("enter standby\r\n");+ r+ @8 J) A' z) _3 B: u
- /* 进入standby模式 */
+ l* Y/ O6 @' C% ~9 j; I b6 L - HAL_PWR_EnterSTANDBYMode();
) l9 f4 s/ v( M0 }5 W4 L - }
0 M/ y* m: D6 W% a3 { - else
W+ F6 y& s; O) q( Z% | - {- \0 i0 ^ h6 V, b
- printf("Parse AT Command fail\r\n");
3 u8 s5 H8 J) Y# r! R% i5 j$ V - }
! I$ A) `8 @$ o5 n, a8 G - }7 q- u+ b W6 l0 C, ^; z
- else
, I; s" U% u$ @ - {
* ^3 t3 }: Z8 C- ]7 B# u - idx++; // 缓冲区索引增加
$ ~9 Q5 P" w: i" h1 l; } - if(idx >= sizeof(buffer) - 1) idx = 0; // 防止缓冲区溢出3 ~% O# w5 H6 ?/ [: R
- }
3 ~9 t/ s- d9 a# m/ e, `4 p - }- r$ F) ~* B- P5 w1 f
- } a/ w1 f4 d, c; b! |
复制代码
$ t; a6 c. r4 g6 Z; a% l+ D V. k% @% F* J
3 ?, t2 V- M* E) o% I* v8 e
- f7 z* i1 z, T; F7 z: c会一直去获取串口收到的数据,如果找到\n就认为收到一包数据,然后去解析,解析成功就刷新屏幕显示日期,然后进去standby。反之继续接收
3 D0 v* z+ Q5 P下面看看解析函数 - bool ParseATCommandAndSetRTC(char* buffer)! k! H; ^; I" _8 l- f% M: h5 A& \
- {) N7 y# ~5 q' r7 U, @4 m( E9 y
- RTC_TimeTypeDef sTime = {0};4 | d9 A5 \3 X$ D- m
- RTC_DateTypeDef sDate = {0};+ h3 ` \1 u. G& ~1 X- s% |
- unsigned int temp_year;
5 Q0 E. F, m8 b" H- {- k - unsigned int temp_month;
- ~& K; J# Z6 Y" G3 e* c' }$ Q - unsigned int temp_day;8 V# e+ c3 F. N/ a9 L" a1 X# G
- unsigned int temp_weekDay;" d( t$ Y2 i% ^, j
- unsigned int temp_hour;
% c! T" |, u- ^6 U9 U - unsigned int temp_min;, c" ^- ^( e3 m$ {: S5 ], u* T
- unsigned int temp_sec;6 h2 }1 x! ~! W9 L5 |' |
- S& }/ G6 f0 U, A
- /* 检查命令的头对不对 */
- L) Y( n# B% N6 X, ~ - if(strncmp(buffer, "AT+configRTCdata=", 17) != 0) return false;7 W, Q6 ~ j0 @
3 \5 D+ ?. I1 ?1 _+ k8 ~ }6 }- /* 跳过命令头部分,再跳过年的前2个数字 */
9 c) o0 u+ m" j4 d$ M4 m - char* dataPtr = buffer + 17 + 2;5 U( u) K% ~& _% x8 v+ n
- 7 Q" u' S. U9 L. K# q+ z
- /* 解析年、月、日... */4 E$ }3 V( h1 E0 s6 z
- if(sscanf(dataPtr, "%2u-%2u-%2u-%2u-%2u:%2u:%2u", ) |3 Z% I8 {$ C* P( X+ U7 v
- &temp_year, 0 [- O, I! L6 b( T; r* T* }: _
- &temp_month,
3 h i( ~- e9 l1 ^( x/ m, x# C! T+ c - &temp_day, 7 q4 K1 K1 V7 \0 p6 ~4 }
- &temp_weekDay,
: w2 g6 O) R' Z4 `8 { - &temp_hour, + R! I7 |# y7 g* \) H8 n
- &temp_min,
; e! ~' Q/ r t, _ - &temp_sec) != 7)
5 P, X# M1 I: A/ K" { - {
2 p0 ^3 `4 ~5 D# s, I& h" U - return false;6 O* i* J' B7 [1 A2 g: q; [! j
- }
% i; U, V+ Z( b/ {) D - 5 Y/ w& `" v9 x2 m; w" z! W
- - h. x4 Y4 l. t4 G: I9 N
- sDate.Year = uint_to_bcd(temp_year);
7 c3 t' t. u- d: @# ?% l4 U/ } - switch (temp_month)
5 \8 i5 J/ l* e# Y9 J e - {
& h: \9 _7 S1 Y* X3 G* \* } e$ _ - case 1: sDate.Month = RTC_MONTH_JANUARY; break;
. b9 Q6 n! D3 Y - case 2: sDate.Month = RTC_MONTH_FEBRUARY; break;! t+ C# R' a! {3 X D) @. u2 W
- case 3: sDate.Month = RTC_MONTH_MARCH; break;; H+ s8 V% w0 O! q2 B
- case 4: sDate.Month = RTC_MONTH_APRIL; break;
# w4 x: _5 R9 c# ~" j1 T( x$ e6 I: ] - case 5: sDate.Month = RTC_MONTH_MAY; break;
1 b0 e. ^; s5 R5 v" M - case 6: sDate.Month = RTC_MONTH_JUNE; break;
2 [5 ~1 a8 x* ? - case 7: sDate.Month = RTC_MONTH_JULY; break;
& g d/ l* K+ E( |( o% a+ O - case 8: sDate.Month = RTC_MONTH_AUGUST; break;
8 M$ t( A; d a( p* ~) }* E- X - case 9: sDate.Month = RTC_MONTH_SEPTEMBER; break;
/ {5 K/ e) n1 Y; N, U - case 10: sDate.Month = RTC_MONTH_OCTOBER; break;+ L/ i5 I2 D: Q, N5 f4 N0 f
- case 11: sDate.Month = RTC_MONTH_NOVEMBER; break;+ }/ u& L, j5 m# I
- case 12: sDate.Month = RTC_MONTH_DECEMBER; break;
) M# f! ` C) p& t - default: return false;# J) C: l$ r" n% V! c6 R4 Y; M
- }
4 ~. ?" Z1 D9 S+ F0 [* }6 F - sDate.Date = uint_to_bcd(temp_day);- G& ^9 \( R: Z' x2 f
- switch (temp_weekDay)
! e0 i3 u0 s+ m6 L7 { - {
% S) T8 ?0 ]+ o! Q! j, Q3 Y# D - case 1: sDate.WeekDay = RTC_WEEKDAY_MONDAY; break;
, o' c+ ^- `& Y( h% ~1 B3 M - case 2: sDate.WeekDay = RTC_WEEKDAY_TUESDAY; break;
. v2 o+ ~( _" u+ A - case 3: sDate.WeekDay = RTC_WEEKDAY_WEDNESDAY; break;
1 L; t7 `0 p: d/ ]3 ^/ \ - case 4: sDate.WeekDay = RTC_WEEKDAY_THURSDAY; break;
" ~; {$ g. j; N, w! T" \- } - case 5: sDate.WeekDay = RTC_WEEKDAY_FRIDAY; break;
9 H* u" W+ m5 O, N# V! { - case 6: sDate.WeekDay = RTC_WEEKDAY_SATURDAY; break;+ C3 Y/ G" {2 S% X2 T
- case 7: sDate.WeekDay = RTC_WEEKDAY_SUNDAY; break;
8 H# p% I" C- T! |, i - default: return false;# l7 [ B' Y$ |& I6 k0 q
- }
6 c* Z7 b, @/ ] y6 A: b4 h
) m7 a9 i' W" G& B# n1 {6 X' |* l( f- sTime.Hours = uint_to_bcd(temp_hour);
! L) ^; I8 u: m7 i1 R4 I - sTime.Minutes = uint_to_bcd(temp_min);
8 ]7 p% a( \1 H - sTime.Seconds = uint_to_bcd(temp_sec);
" F2 y4 b, @( _, S) k1 }$ x - sTime.DayLightSaving = RTC_DAYLIGHTSAVING_NONE;1 x) j% c) X2 w1 q9 ?2 v+ M
- sTime.StoreOperation = RTC_STOREOPERATION_RESET;
& g* r8 Y1 A7 M o& b/ \ - 0 X+ R0 k0 r& ? S
- set_rtc_data(sTime, sDate);
6 u) U3 h/ b6 U. l" r) v
9 O4 s" k) a0 o1 @( T$ r; C- print_rtc_data();$ P+ X4 _# u- Q- B6 K& H1 L
% m) }4 f/ m- W* ]0 n/ v% V. M; [$ K8 V- return true;
, T5 }- w) ~" G- T7 r4 Y - }7 y# d. Q0 ~1 S2 S' l
复制代码 \+ y2 a, ?; G% `. X
如果解析成功就设置RTC,反之返回false。 : R: x9 U0 V4 ^/ e# f9 N% F
i. c! E5 P8 `
我输入的串口数据是十进制的,但是RTC使用的是BCD,所以需要转换一下,这就是uint_to_bcd函数,他的内容如下 - uint8_t uint_to_bcd(unsigned int num)
* A9 J( l" P% x: ?! K3 @6 n0 Y, Q5 A - {
& q! K) ~: ~4 N - // 检查输入范围,确保0-99之间1 ^% b4 `8 c2 s- E* U
- if (num > 99) ! z1 Y$ F+ ?2 p6 @
- {! P8 M a" O+ w9 P" @
- // printf("Error: Input out of range. Maximum 99 allowed.\n");
, S- }* O) L4 J- ]" o - return 0; W; F' ]) M" z, {1 ~' u1 M9 `
- }
2 Y; r @7 L+ g( w) _* {8 ?
# g! ~7 Y( e$ b% D9 l1 g- // 将十进制数转换为BCD; K3 J9 m" w, L/ n# `
- uint8_t bcd = (num / 10) << 4 | (num % 10);
& l) x: v4 F9 g% a8 f. O - ) n: t2 R; [2 |. a4 R
- return bcd;- j- i+ m, I. s! L
- }
5 u2 `- Y& q9 {: O; T4 ~
复制代码
) q3 O. ~3 F# g/ b/ F" f4 V3 i核心的逻辑代码都在这里了,屏幕显示的函数与具体逻辑无关,我就不展示了,可以自己去看附件中代码 . U2 [% J# s" |* G! R
]& f2 m* B0 Z& y; z8 l
四、上位机注意事项
# U3 o8 G4 [# t1 z上位机软件是用python+tkinter+serial写的,所以保证你的电脑上安装了python环境(我这边是最新的3.12.4),然后要安装tkinter、serial的库,否则运行时会报错:“ModuleNotFoundError: No module named ‘tkinter'”或“ModuleNotFoundError: No module named ‘serial'”(我忘记是serial还是pyserial,大致就是这么个意思) + K8 l* O Z0 Q
+ B6 Z! p9 t' b
- d* A9 T* s+ g8 v你可以用以下指令查看是否安装这两个库 如果安装了,会弹出一个窗口,这是tkinter的一个demo 如果安装了,会打印 出可以用的COM口 ) L8 p3 h/ h! L
如果你没安装,可以用以下命令安装serial tkinter一般都是在安装python时自动安装的,如果你没有可以看一下这个 文章
3 ~# E- d7 u3 }* e) `6 W/ [然后上位机我也提供源码在附件,如果EXE实在无法运行,可以尝试用VSCODE调试运行,我这边之前就是调试可以运行,EXE无法运行,需要安装东西,很奇怪
$ d) Q2 w8 e7 q然后便于测试,我提供了强制写入23点59分30秒的版本,可以通过注释以下代码来实现 $ J+ S4 M+ e( D8 p+ R% I5 `- L( u0 t
: ^5 y# A b/ a1 o
! g5 g9 M3 L4 m q& I& E, L% c* M- a8 |0 L# Q! X
五、后续改进想法 ! V/ U5 n+ G9 l
这个demo现在其实处于一个初始状态,低功耗部分还没有调试过,目前已知的硬件上肯定需要做调整,否则功耗挺高的 / u( y9 M3 a1 A
1.LED3这个灯要干掉,他一直亮着耗电 0 B0 q9 @/ E" r3 C0 n
2.供电及串口,之前我测试standby功耗是是选择使用CHG跳帽的,现在为了可以借用STLINK的VCP被迫选择STLK,这会增大功耗,后续我打算使用外部的串口,供电改回CHG。板子上需要把SB48/45焊接一下,这样串口可以引到arduino的D1、D0口上,配置完时钟后拔掉接线
2 {# j W8 i2 H5 \" S2 j 3 Q, Q) P3 ?) I# S) A1 W
6 a- p" A" Z6 h$ h
$ z" W( f# D( v/ e3.STlink VCP的R23、24电阻干掉,我一直怀疑MCU会有电从这个漏过去,导致之前测量的功耗高,反正现在打算用外挂的串口,这个我就直接干掉。
" c0 O R$ h# F+ h, D1 w# a# h4.配置RTC的机制有点不灵活,用户想要配置RTC,必须断电再上电,不友好。毕竟RTC时间长了就会偏,手动重校时是必须的。后续可以把板载的用户按键利用起来,把它作为唤醒源,当按下后,MCU WKUP,等待用于输入配置指令。多久没收到正确的指令就再进入standby
+ U" B. U+ K; S y5.美化一下界面 5 d/ j3 F/ f, Z! y1 t6 P, E
6.增加wifi模块,这样可以实现每天网络校时,并且还可以获取天气预报信息,让屏幕看上去不那么空 + Q% v& C( z6 Q3 z( S' { J$ f
7.如果有合适的液晶段码屏,可以加上,用于显示温湿度数据,躲开了电子墨水屏的问题
3 F; Y5 P' ^3 q6 H: c2 {
' g' a. ~9 r6 }( E* q! v4 X& G
2 V8 G4 }6 E5 a5 X3 {6 a
7 E, { D$ F. m感谢各位读到这里,如果你有更好的想法或者有疑问或者代码有错误,欢迎在评论区交流 ' \/ A9 `) N4 Z- v k# _- t, f! J" e0 n
4 `; M- ~1 x9 I M: ?- z
7 R" d1 f' ]3 J2 J; A9 q! P( R! C* W
8 g$ z9 M: ?2 F6 N R* b
9 M* z2 O" A' D
项目源码及上位机如下
3 P! I9 q1 D& X |