本文是STM32U0测评系列的最后一篇。我认为STM32U0最吸引我的就是他极致的低功耗,ST官方也说他非常很适用于水表等产品的应用,于是我只做了一个电子墨水屏RTC日历。; {8 K3 V# N; V2 a2 V. {
) i- f: Z& B! d O5 S$ L
, h) v( I2 X0 _# e
一开始我是想做日历+时钟+温湿度。但是很可惜我手上的这块电子墨水屏不能局部刷新,只能全刷,而全刷要15s,也就是说1分钟内只有45s是正常显示的,这也太鸡肋了,完全没有实用性可言,所以最终决定就只做日历,一天刷新一次,功耗也是嘎嘎低
+ Y) n% M4 v1 c3 ?) I9 Y, }- L5 N# W. K O/ s: ~4 i3 L u
) X! ]0 D; |; l1 t" q* D由于中途有很多调试工作,我就不再一步一步讲解是怎么做的,简单讲解一下demo功能、接线、代码逻辑、上位机使用注意、后续改进想法等+ \4 E" G- a) ^6 r! m" W* |
2 q& f1 Y( s2 Z! B1 g* K
- C% J Z5 ?6 r: q2 T: w一、简述功能
! y" l* M' a) y/ q9 S使用STM32U08 NUCLEO开发板外接一块我自己做的电子墨水屏驱动板(软硬件已开源,地址:https://github.com/BUYITAO/My_E-Paper_Driver),显示当前的日期,每天24点整由RTC闹钟唤醒刷新屏幕,平时MCU都处于STANDBY,这样可以让整个系统处于最低功耗状态。正常显示如下(偷懒了,没有做界面美化设计,请见谅)
7 `! l& y# ~0 d$ k. \: G0 t
9 g9 C% O# }* g; `6 x6 C! d, S
% ~! B4 z# `. S) B, [) S6 x
系统首次上电后由于RTC参数丢失,会处于默认的预设状态,需要用户手动设置一下时间,此时电子墨水屏显示“Please config RTC data”,如下图所示
. U4 N: d4 Q) r4 l. ?$ M# I5 L # D& i% A( q( S2 L6 ^! N9 _: D
+ V& B5 A0 `/ }' R, |
- q8 D0 ^* m: _6 j/ B, c: U) x用户可以通过串口发送配置参数(这个我做了一个上位机可以一键发送配置指令),MCU收到后,刷新屏幕,再进入standby。上位机界面如下图 9 J* L& _$ e! s% s8 `& s* B
+ [, e4 ]( G" c- r, b1 n; U
$ F( o7 O q J+ F) {0 V3 P: U
( d' Y2 X* d8 M; H7 A0 p8 d( G9 e4 x" A/ ]1 ? q
- ?- ^/ x1 O; Z- D9 p& I3 W二、硬件连线 " \! ~# l1 G% Z0 ^
/ ^9 J( X4 }) e9 J6 M6 F1 u( x
" g! Z, Z) [) F% k1 m; g
( I) Q+ R/ B5 J! q, Y0 M/ H( i墨水屏驱动板 STM32 # ~% ?" ^7 J2 e. s
MOSI PA7
0 D) v/ m" ]( w# p# tCS PC9 ! f: }. o5 r+ u ?; n2 X7 I2 J
RST PC8
$ U. J3 Z$ `; c/ |CLK PA1
' |! ~5 U; Y# y) T- s! u- `DC PC6
( W4 a7 m8 B1 w( PBUSY PC5 % s( L5 J# Y: v) i! i
0 D4 {, D8 h' o, D
0 Z& w( w V e4 _照片不太看得清楚开发板上的接线,我把原理图放上来
/ u1 {& t7 k2 a4 l # P" W% C8 w6 D
' g+ H$ C! Z6 }
7 j. }2 M% s- O然后串口部分我借用了STlink的VCP(这个最后改进点部分我会讲到,用这个其实对功耗影响挺大的,后续打算改掉)
3 X0 I+ E D6 L! w% X2 S& d# `
& P" @6 K: x8 m0 [; |0 L2 b# @: @: h: s; y0 W1 ]
三、代码简单讲解 * l" r& g9 x; L6 x
, l; L& P0 \ e& ? y
CUBEMX打开SPI,然后配置了GPIO。这样就可以实现与电子墨水屏的通讯 / E& V# ?. X; |/ u
+ [, Z* R3 Q7 {- G( W! I
$ O& w! P/ R0 l S3 a* i
" z7 {6 }7 y0 i1 v
: R& [7 K6 j6 e
4 _6 U6 x' X V3 \9 T + l2 \- K2 t% |) \- ]# o# S$ f
然后把GPIO和SPI的库改成LL库(个人习惯,这些基础的外设以前LL库用惯了,毕竟可以看代码学习对应寄存器的操作),然后生成代码即可 - p1 S1 z" @2 b c
, ~' j8 q4 V* e2 c, f! U ( h- B/ v' i9 ^& @+ Q( }$ a$ h% g
***这里需要注意一下生成的代码有两个BUG,毕竟现在U0的PACK才第一版,有点BUG很正常,希望ST可以看到我的这个文章,在后续的版本中修复一下
, [+ O: o" @: k" B第一个是在main.h中,红框这个没有换行,会导致编译报错 # \; Z5 Y( H* r5 b* B! N
5 r9 S( p! t9 I( s- B3 [: Q2 F
5 @* p2 D: K7 }' @$ @- H6 Y% ?
第二个是生成的工程会强制设置为V6编译器,V6编译会疯狂报错,改成V5就好
) g+ I' O, ^% ]* F! Y- T5 ?
, X8 J. A, s2 [* n1 ~
8 t2 p4 m+ k9 E1 ^$ @: F
6 c/ Y# D% W; S- D" I! u2 t, ~ |) U& C: C/ z# Q, F _- V( b
: m& W! X! h2 H( Y然后关于电子墨水屏的移植、配置啥的我就不细讲了,在我分享的github上有详细的readme。接下来简单看一下功能代码
# b6 ~& t2 G7 p& ?while1前我会初始化墨水屏,然后判断是否为首次上电。如果是首次上电,就让墨水屏显示“Please config RTC data”,反之就清除标志位,刷新屏幕(到这里就是RTC闹钟到时间了,要刷新屏幕,显示新的日期),然后再进入Standby - /* 初始化墨水屏 */+ v* Z9 {5 d1 a5 i$ \' {
- E2213JS0C1_Init(0);& Y4 i6 c" }/ i6 h* [) |9 A
- /* 是否为首次上电 */
* J% B: [7 u% v7 V, ]. h8 K- P - if(__HAL_PWR_GET_FLAG(PWR_FLAG_SB) == RESET)* k L- p0 G$ s* N3 F4 E4 K6 r) v
- {
' f) E1 [, O5 Y8 w; e! R& U - printf("normal run\r\n"); x9 S2 i9 x3 h" `" Z* a l
- /* 打印RTC时间 */
- U" b- S: Z b3 J( T1 T7 y - print_rtc_data();$ N5 L) R6 R% I% [
- E_Paper_show_first_power_on_page();9 @* I% ^& E1 Y$ V1 }& N3 Y
- }7 S8 _- Q; x( U7 z/ t
- /* 从standby唤醒后的复位 */
, y* Z: }( V* o3 p; D - else/ b! s5 @" o- K+ B/ g. ~: v
- {) g7 P5 s' W4 H" k
- printf("wkup from standby run\r\n");
: s" {8 @4 y' L! E - /* 清除standby的标志位 */
" u8 X0 F( G" _) k - __HAL_PWR_CLEAR_FLAG(PWR_FLAG_SB);6 T" s( s( t' X9 E* l* ^* q
- /* 清除闹钟标志位 */
7 t) M U l6 `3 [! K5 t - __HAL_RTC_ALARM_CLEAR_FLAG(&hrtc, RTC_FLAG_ALRAF);
" [8 u- T8 ^% d - /* 刷新屏幕 */
8 p% E- Z+ c, J3 d+ m3 D - E_Paper_show_calendar();6 X- R# ?/ }4 Q8 K9 m9 d- ?+ p3 {. w6 F6 K
- /* 进入standby模式 */+ g" o1 t2 P/ S* I
- HAL_PWR_EnterSTANDBYMode();
4 }0 J$ Q$ L/ {3 E0 y$ I - }
复制代码 * q" L3 O- U/ X6 I5 H" [
- i* z* t+ o* ?. o6 r) y: M+ O6 |1 d$ W) d8 ~6 [
如果是首次上电的,可以看到没有调用standby函数,所以他可以向下进入while1,while1中的函数如下 - if(HAL_UART_GetState(&hcom_uart[COM1]) == HAL_UART_STATE_READY)$ _% f$ u( I3 l2 o2 t$ a9 r& R
- {
+ _ q. U3 l( U$ B+ p/ D - if(HAL_UART_Receive(&hcom_uart[COM1], (uint8_t*)&buffer[idx], 1, HAL_MAX_DELAY) == HAL_OK)3 }* x6 b N" A! i
- {( X! N$ ?; q& k5 l y
- if(buffer[idx] == '\n') // 假设\r\n'作为结束标志
0 [5 p, q! d' C" D" i - {3 J/ m4 k1 S3 `2 s1 O6 x
- buffer[idx] = 0; // 添加字符串结束符4 g* b( z+ u u$ q# ~
- idx = 0; // 重置缓冲区索引
: M& Q( W) \. o9 e - if(ParseATCommandAndSetRTC(buffer)). N. b) d1 c2 Y8 W
- {/ L) c/ @5 W4 I9 b
- printf("Parse AT Command success\r\n");$ R2 d2 b7 P8 e/ K
- /* 刷新屏幕 */ B3 j5 P7 \( ?+ q' @* G
- E_Paper_show_calendar();6 o! l' w. z1 _7 ?" \* G% B m
- printf("enter standby\r\n");
! K' k. E& R9 L - /* 进入standby模式 */
1 h% f4 f1 P; _' s; |2 ? - HAL_PWR_EnterSTANDBYMode();, Q! F5 f# F- {% t" j
- }
3 {/ u1 d0 o$ E% f - else1 j1 f0 Q- B2 m' V
- {
( u+ c" q% E1 [9 Y - printf("Parse AT Command fail\r\n");* p- E* i6 t- J9 j- Y& |, F
- }
& L6 y( N. [: A a% l n2 A - }: q+ ?' c% A3 |% ~
- else! v- I& f9 R2 ?0 R& n7 G
- {
/ o; U7 [( L8 Y* u& N* @ - idx++; // 缓冲区索引增加$ r7 |3 ^; y. @' u8 }
- if(idx >= sizeof(buffer) - 1) idx = 0; // 防止缓冲区溢出3 J2 C* w4 {' @) [# w
- }
" j. `% I) L# ?% A/ q8 y- ] - }9 A9 [2 ]/ p, T: a( w& }
- }* L. g5 z' x( p' }
复制代码
: _2 l4 s+ H+ g: q0 T$ Y- F. {1 ]8 K; r: Y5 c
# d' w. X* H' F* ~. j/ b/ o4 _" r$ u, U/ u* u$ ]: }& V5 G H8 S
会一直去获取串口收到的数据,如果找到\n就认为收到一包数据,然后去解析,解析成功就刷新屏幕显示日期,然后进去standby。反之继续接收
5 S" r6 z& Z: N$ h' c3 P& K& |下面看看解析函数 - bool ParseATCommandAndSetRTC(char* buffer)5 C5 g# [) d y3 a
- {+ A% I0 N1 E- W9 e+ v
- RTC_TimeTypeDef sTime = {0};7 I# \0 |* K9 o3 Z/ W
- RTC_DateTypeDef sDate = {0};
2 _& R1 T0 n' n - unsigned int temp_year;7 o9 `: Z/ m" x9 Q4 C
- unsigned int temp_month;; a9 l3 R9 K. @5 q' Q6 M1 w, B
- unsigned int temp_day;
' \" X: a2 Y; W1 Z) a - unsigned int temp_weekDay;
1 Q6 o; N& |' B! b - unsigned int temp_hour;
1 A+ K: y- Z0 T) C2 f - unsigned int temp_min;
5 f' S; N- x# x$ k7 p1 ? - unsigned int temp_sec;
1 [% b; F" \$ A0 n% ]0 N) Y
* z- J5 o, c# p8 M- /* 检查命令的头对不对 */
+ p9 P8 Y( A: R' ^$ r* E: N, X - if(strncmp(buffer, "AT+configRTCdata=", 17) != 0) return false;
4 } [- U; |5 j/ j+ v: L! Y
& o+ z( A+ x) }0 L q# B1 l0 h- /* 跳过命令头部分,再跳过年的前2个数字 */
. a7 E! E$ ~( _: [! o5 l - char* dataPtr = buffer + 17 + 2;5 m# t' Z2 B6 ~. w# n0 n$ ?4 ^- k5 q
- / v( {& X% h/ f, y R& o- q
- /* 解析年、月、日... */
8 \( X9 K( V* I& ]! B - if(sscanf(dataPtr, "%2u-%2u-%2u-%2u-%2u:%2u:%2u",
( E7 f5 Y* w+ i0 |4 `! ~) s- ] - &temp_year,
# |/ x9 P& g+ z6 u" l5 ?3 E - &temp_month,
: i# d; b9 v6 l2 n2 e2 B - &temp_day, 4 w. C! |5 x! } O: E4 o
- &temp_weekDay, 3 c' ~( x2 I6 {. F
- &temp_hour, " @0 s w2 W& Q# \% A; I% R% h
- &temp_min,
+ j. Z3 w) B% ?2 i* ^6 i - &temp_sec) != 7)
7 z' \/ w& k1 T9 t0 W* {' g - {
. @! z8 W( V J; H; P; K6 l# ? - return false;/ f$ c, i& x; ]/ k- G
- }
# u3 C; h' k6 z& p2 {$ ^3 @& P/ Z
2 x+ u4 j K7 n3 h7 T- S# `3 N
2 j6 C0 V4 k/ p9 o* v7 ]5 i: ^+ X- sDate.Year = uint_to_bcd(temp_year);
2 \9 t( s) c. [1 X8 h - switch (temp_month)
8 Q [ }' D5 q$ O" ]' S9 ~ - {
, ~) m2 q3 T& I7 U; H' M. b0 y - case 1: sDate.Month = RTC_MONTH_JANUARY; break;
$ k6 ~, s+ I6 r% L6 a/ b* i, { - case 2: sDate.Month = RTC_MONTH_FEBRUARY; break;$ M+ f- `! X' y) D8 Q0 ]+ c
- case 3: sDate.Month = RTC_MONTH_MARCH; break;0 S! E; o- c U9 y. d# d& @
- case 4: sDate.Month = RTC_MONTH_APRIL; break;
! j, m; I L% j; o* c& Q! Y% P3 R D - case 5: sDate.Month = RTC_MONTH_MAY; break;* z, J8 P9 [7 {8 [. [) t1 t% S
- case 6: sDate.Month = RTC_MONTH_JUNE; break;
7 _ t5 B& I) L! Z: L8 d - case 7: sDate.Month = RTC_MONTH_JULY; break;
}; i- w& b: M - case 8: sDate.Month = RTC_MONTH_AUGUST; break;
; c* j3 _* j, |" _) y2 L - case 9: sDate.Month = RTC_MONTH_SEPTEMBER; break;
2 b- [ j3 E" ] } - case 10: sDate.Month = RTC_MONTH_OCTOBER; break;: h# v* K/ }1 I9 n
- case 11: sDate.Month = RTC_MONTH_NOVEMBER; break;, c- R- k2 j4 Q
- case 12: sDate.Month = RTC_MONTH_DECEMBER; break;' G" M: R* A3 J5 z3 K
- default: return false;3 z+ z3 o% I# O; j$ }
- }* K' ?7 d; E+ V- y2 p6 _; p
- sDate.Date = uint_to_bcd(temp_day);
, N! Z6 c3 b# l$ C - switch (temp_weekDay)
! s& Y4 c8 L3 ?" _, p& \$ m- y9 } Y - {& K4 M0 _9 i [
- case 1: sDate.WeekDay = RTC_WEEKDAY_MONDAY; break;% @: o) ~: r" f% c
- case 2: sDate.WeekDay = RTC_WEEKDAY_TUESDAY; break;
% w/ s% |- R' u* x7 Q1 d3 n - case 3: sDate.WeekDay = RTC_WEEKDAY_WEDNESDAY; break;
, T: Z+ N3 W) f& a - case 4: sDate.WeekDay = RTC_WEEKDAY_THURSDAY; break;
: b- I6 @$ z& r5 X+ K - case 5: sDate.WeekDay = RTC_WEEKDAY_FRIDAY; break;
7 D# E4 g# W4 J. ?) o4 y' N - case 6: sDate.WeekDay = RTC_WEEKDAY_SATURDAY; break;1 i. Z7 h9 g3 K2 v7 D/ I+ }
- case 7: sDate.WeekDay = RTC_WEEKDAY_SUNDAY; break;
/ y/ u" k! b* g$ V4 R - default: return false;& s$ y( x; C2 @0 ]+ ]0 n
- }# W. Z8 @1 T! ]/ ]* p/ \' z
- $ t, W8 O8 s5 b6 g1 L( A
- sTime.Hours = uint_to_bcd(temp_hour);
1 Y+ H0 O( p- F# {2 J - sTime.Minutes = uint_to_bcd(temp_min);
9 @1 W! h; [) z, s) j! S$ t - sTime.Seconds = uint_to_bcd(temp_sec);
, S' w* I5 E% F1 N - sTime.DayLightSaving = RTC_DAYLIGHTSAVING_NONE;" B! k: X& ^" y& \9 P; O' W4 V
- sTime.StoreOperation = RTC_STOREOPERATION_RESET;
7 ]) {2 `! J: }5 e, ~' W
4 [7 k& r& D, ]8 t; |4 X) Q, c- set_rtc_data(sTime, sDate);
}2 |% `- a: r# D
, c, C! s4 V* L- print_rtc_data();# w8 X" B3 w2 d
- ( T2 U( y- _- {% @
- return true;( |& h) S- M6 Q
- }, F$ Y0 ^0 a) A) Z2 \6 v2 z$ S' j) y
复制代码 9 Y9 K4 r1 v* k8 h; ~9 t
如果解析成功就设置RTC,反之返回false。 , W" @/ r' d5 f6 w7 F1 `7 u
, d/ i3 v. Q% d
我输入的串口数据是十进制的,但是RTC使用的是BCD,所以需要转换一下,这就是uint_to_bcd函数,他的内容如下 - uint8_t uint_to_bcd(unsigned int num) 7 E s7 y8 n+ ?( Y \
- {5 F. W. n" s! G3 x7 A
- // 检查输入范围,确保0-99之间
& O% I3 |' B8 q0 \+ m - if (num > 99)
+ E% x9 j3 ^( Y: L* W3 D. k - {
3 i& C" p4 X8 J* [1 h - // printf("Error: Input out of range. Maximum 99 allowed.\n");
, l, a$ t( I5 a( K9 ^ - return 0;
4 U8 n8 l" B/ R - }
9 B# _- p3 E. j H8 U2 O
! t- ?" G1 R/ c. V- n2 F3 D- // 将十进制数转换为BCD
: M+ G) b5 c1 d& N+ K9 {+ R - uint8_t bcd = (num / 10) << 4 | (num % 10);# G6 p% j; G1 U0 p8 s3 u
- ! [; }7 v# C, q! m9 i: G z9 v
- return bcd;
5 E# e: n7 R4 V) d( ^7 Y - }/ Y1 i+ b( ] @# G! @5 O3 e
复制代码
( @$ ]3 D9 r8 l" z. I0 d核心的逻辑代码都在这里了,屏幕显示的函数与具体逻辑无关,我就不展示了,可以自己去看附件中代码 " K% g- Y4 m% n0 W
9 d$ b9 m. A* i' A, e四、上位机注意事项
/ ]+ U* {5 h5 E k0 o. w上位机软件是用python+tkinter+serial写的,所以保证你的电脑上安装了python环境(我这边是最新的3.12.4),然后要安装tkinter、serial的库,否则运行时会报错:“ModuleNotFoundError: No module named ‘tkinter'”或“ModuleNotFoundError: No module named ‘serial'”(我忘记是serial还是pyserial,大致就是这么个意思)
6 M+ B2 Q+ ]& s7 L; L/ }2 c
% Y! [; E, @& W3 \) | e5 L. k
: r+ X; f; Q+ u3 i" M& ~. z0 f) j" B你可以用以下指令查看是否安装这两个库 如果安装了,会弹出一个窗口,这是tkinter的一个demo 如果安装了,会打印 出可以用的COM口 2 G( Q6 b+ j) d# q1 C! W$ [4 b. I
如果你没安装,可以用以下命令安装serial tkinter一般都是在安装python时自动安装的,如果你没有可以看一下这个 文章
+ X. e, ^& f6 e( @然后上位机我也提供源码在附件,如果EXE实在无法运行,可以尝试用VSCODE调试运行,我这边之前就是调试可以运行,EXE无法运行,需要安装东西,很奇怪
% ^. G$ n9 t" M! Y9 c然后便于测试,我提供了强制写入23点59分30秒的版本,可以通过注释以下代码来实现 1 M3 u8 b) V$ j1 q5 u
5 `8 Q. q9 P( L& `% H
% f/ f( {) B1 ?1 B6 ]. c8 X
8 U$ t' x/ m# o6 }' w. L五、后续改进想法 3 {+ h1 j: C1 `* U, n. |' d
这个demo现在其实处于一个初始状态,低功耗部分还没有调试过,目前已知的硬件上肯定需要做调整,否则功耗挺高的
+ H' H0 }. ]+ G$ I: t; @1.LED3这个灯要干掉,他一直亮着耗电
8 y2 t2 {& q4 Z9 m! t6 E2.供电及串口,之前我测试standby功耗是是选择使用CHG跳帽的,现在为了可以借用STLINK的VCP被迫选择STLK,这会增大功耗,后续我打算使用外部的串口,供电改回CHG。板子上需要把SB48/45焊接一下,这样串口可以引到arduino的D1、D0口上,配置完时钟后拔掉接线
7 ?8 l& O% e5 n9 ?. q5 m6 K0 H
) Z p4 C, o: E4 ]0 r" C4 l
) h8 h3 B+ p y: d
2 m1 R0 E+ X2 O* a Y3 Z: l3.STlink VCP的R23、24电阻干掉,我一直怀疑MCU会有电从这个漏过去,导致之前测量的功耗高,反正现在打算用外挂的串口,这个我就直接干掉。 8 {2 ?7 j; o" W9 K+ V" c' h
4.配置RTC的机制有点不灵活,用户想要配置RTC,必须断电再上电,不友好。毕竟RTC时间长了就会偏,手动重校时是必须的。后续可以把板载的用户按键利用起来,把它作为唤醒源,当按下后,MCU WKUP,等待用于输入配置指令。多久没收到正确的指令就再进入standby
1 {; _3 T7 [3 K! c* d; c5.美化一下界面 % k) Q5 t+ `% R' l# @6 p
6.增加wifi模块,这样可以实现每天网络校时,并且还可以获取天气预报信息,让屏幕看上去不那么空
$ P2 d* r) Q8 C: f- o) t3 L9 O! d7.如果有合适的液晶段码屏,可以加上,用于显示温湿度数据,躲开了电子墨水屏的问题
7 E0 g# \$ D! P) {, B7 d5 l T! m! `# \
6 Q1 C6 R/ @& x- o
; R; X" W9 ~( w- Z3 N: k/ T$ n9 D- @& p. f7 N: g( H
感谢各位读到这里,如果你有更好的想法或者有疑问或者代码有错误,欢迎在评论区交流
/ o+ F4 R0 {3 ~$ W# K m9 z; j4 g
/ A. A: X2 a' i; x1 i/ n5 \2 b
+ `* t/ |! H2 @" h8 w2 S& u1 n8 [9 O' i( Z. h2 `
8 z3 @9 d' t) F$ O% E3 k6 _1 j( Z
项目源码及上位机如下 / ~0 B6 a$ X) R6 Z$ t
|