本文是STM32U0测评系列的最后一篇。我认为STM32U0最吸引我的就是他极致的低功耗,ST官方也说他非常很适用于水表等产品的应用,于是我只做了一个电子墨水屏RTC日历。
# d& w5 l, U( W- G8 p
' O7 l) c/ V" o( \$ {* Q& p! R( Q' j, {) i1 Y
一开始我是想做日历+时钟+温湿度。但是很可惜我手上的这块电子墨水屏不能局部刷新,只能全刷,而全刷要15s,也就是说1分钟内只有45s是正常显示的,这也太鸡肋了,完全没有实用性可言,所以最终决定就只做日历,一天刷新一次,功耗也是嘎嘎低4 J9 l( M1 {4 {. l5 J8 b
) z7 j% e$ B' h8 R3 v) L
) W* E5 y4 Q9 B/ D1 ~ O
由于中途有很多调试工作,我就不再一步一步讲解是怎么做的,简单讲解一下demo功能、接线、代码逻辑、上位机使用注意、后续改进想法等
3 Y/ l% X0 y3 Y* F2 h2 ?3 C2 a- E- ^9 I: l
8 O0 D3 A j" ~
一、简述功能4 i' n+ O M4 x% `) u
使用STM32U08 NUCLEO开发板外接一块我自己做的电子墨水屏驱动板(软硬件已开源,地址:https://github.com/BUYITAO/My_E-Paper_Driver),显示当前的日期,每天24点整由RTC闹钟唤醒刷新屏幕,平时MCU都处于STANDBY,这样可以让整个系统处于最低功耗状态。正常显示如下(偷懒了,没有做界面美化设计,请见谅 )
* C( Q5 ?5 D5 W3 U6 D2 h C# k3 U" {# A* [
# Y- b$ e9 e' }2 v- `3 ]: d' M
+ L; `4 z9 ~: v3 v; `6 E$ P系统首次上电后由于RTC参数丢失,会处于默认的预设状态,需要用户手动设置一下时间,此时电子墨水屏显示“Please config RTC data”,如下图所示
- C# ~( Y$ ~1 H) J0 }- ^ 1 f* W9 N3 q0 l) {+ o: x+ o& H9 @
E/ n% Y6 a. X
# U* }3 \+ }% Q% }用户可以通过串口发送配置参数(这个我做了一个上位机可以一键发送配置指令),MCU收到后,刷新屏幕,再进入standby。上位机界面如下图 , i# t- u/ S& m" Y
5 k2 d* z' d' G1 w( R9 ]. a
1 W$ ]/ W9 C' q$ Y
7 n" M; X/ n: d I6 p* p5 o, Y5 Y/ n5 ?9 G
4 u/ l( n( {5 N2 I8 {3 X8 r
二、硬件连线 ; t Y* t# P, H: R( Z# g
3 C0 P: N% c6 v+ x
1 [$ ?: z* V* k
9 j% ?1 m: \8 L) M3 y墨水屏驱动板 STM32 7 ^( q& ^. b1 i% C
MOSI PA7
/ k. [5 }# x$ O$ A' I2 FCS PC9
) z7 k* [, z# rRST PC8 4 R& T, T. D7 C5 V2 Z8 M
CLK PA1
8 Q: H. W' j {9 N0 {3 C9 `DC PC6 ( E& n4 [2 e& r4 p7 s6 z0 {5 a, j
BUSY PC5
4 e6 T6 Z% m5 i( f# N4 B3 r* `
0 w- q1 L& M, ~+ U; i% E& h( n( m% F8 r4 ?
照片不太看得清楚开发板上的接线,我把原理图放上来 7 G! @8 H1 a& K6 K* g. I! ^1 C
2 m/ D) `5 N. c4 U. f
) U9 b. I: J1 _5 [% m2 M
' D4 `, q& n( b然后串口部分我借用了STlink的VCP(这个最后改进点部分我会讲到,用这个其实对功耗影响挺大的,后续打算改掉) ' w2 W4 D7 }0 f& I
5 F8 e4 i% u' Y1 N r9 T" j1 N$ V, O7 j- D' h% N7 e
三、代码简单讲解
" v: g) c4 ?, y
8 f+ }" i5 e& ~CUBEMX打开SPI,然后配置了GPIO。这样就可以实现与电子墨水屏的通讯 9 J$ ?* C+ T( @5 J* T0 n a% F
/ Z! D. x, ` }1 N& [
5 k$ ?* O2 g F* |) P; Y$ c; S A
' {* K* p* ?* V9 C2 H! L$ F% \: q: K0 o
+ J) B0 {' Y: }5 K1 z
8 Q! k- Y: W" b7 }5 M J然后把GPIO和SPI的库改成LL库(个人习惯,这些基础的外设以前LL库用惯了,毕竟可以看代码学习对应寄存器的操作),然后生成代码即可 # f9 `5 x6 C$ |* Q
9 N/ l6 w2 t8 u4 ? Z # b8 m7 Z# i8 t4 V, w
***这里需要注意一下生成的代码有两个BUG,毕竟现在U0的PACK才第一版,有点BUG很正常,希望ST可以看到我的这个文章,在后续的版本中修复一下 p9 ^6 V- w( s1 k& \
第一个是在main.h中,红框这个没有换行,会导致编译报错
7 R, } I. d) n; U7 e# J
: I/ S8 d; u& k; h a/ M/ ]+ s! _7 U
第二个是生成的工程会强制设置为V6编译器,V6编译会疯狂报错,改成V5就好
; \( p3 f9 u! H) e- [, I7 h
5 T- B8 w5 L' b
$ `( X) e! z$ X$ w, v( Q
( A% B5 S% @7 s) _
0 E/ t; o: f: U: N/ w4 l5 O8 z! h/ K5 w/ F" A# d6 H
然后关于电子墨水屏的移植、配置啥的我就不细讲了,在我分享的github上有详细的readme。接下来简单看一下功能代码
" Y$ N4 N/ h, d% ^! B( u8 Bwhile1前我会初始化墨水屏,然后判断是否为首次上电。如果是首次上电,就让墨水屏显示“Please config RTC data”,反之就清除标志位,刷新屏幕(到这里就是RTC闹钟到时间了,要刷新屏幕,显示新的日期),然后再进入Standby - /* 初始化墨水屏 */
; P# n+ K1 w+ b* [! R - E2213JS0C1_Init(0);9 c- i" H% `4 T5 m' M; A3 Q+ \4 z
- /* 是否为首次上电 */
8 [' j$ P( k- A( ~3 a. |, T) c - if(__HAL_PWR_GET_FLAG(PWR_FLAG_SB) == RESET) m, Q0 K6 `1 K1 y& O# E7 E
- {/ w2 B, A h6 O% N/ h7 a0 T
- printf("normal run\r\n");9 L1 L, T7 i; i' y/ h9 b' c7 `+ Q
- /* 打印RTC时间 */
7 q( Z* b! ^$ x5 R& H9 z( O0 u9 ] a - print_rtc_data();
- d! l9 ^$ u$ F& h1 K - E_Paper_show_first_power_on_page();
. u# z: c8 s) r7 [6 V - }5 B2 g: u7 n1 @3 E$ Q, ?! w8 ?
- /* 从standby唤醒后的复位 */
: X3 V3 ^3 P6 E: @: X2 W - else
9 }8 l1 [( Y1 M f& H4 k" O3 c$ @ - {
! _( G9 o, x! c1 X# Y - printf("wkup from standby run\r\n");
5 J8 j9 b& T S7 M: j" g0 g! w - /* 清除standby的标志位 */
d& v* W9 w2 o& e - __HAL_PWR_CLEAR_FLAG(PWR_FLAG_SB);, }7 I" @- T/ ]$ g2 p, m
- /* 清除闹钟标志位 */
, A; g. L8 ~' b- i. M, a - __HAL_RTC_ALARM_CLEAR_FLAG(&hrtc, RTC_FLAG_ALRAF);, O' W5 f$ @4 z2 [
- /* 刷新屏幕 */; t9 R- _5 O) d6 }* Z, f
- E_Paper_show_calendar();: [5 w7 O; O5 D6 J
- /* 进入standby模式 */
4 Y- A8 U& ]) @" f5 Q# n; Z - HAL_PWR_EnterSTANDBYMode();! v, ^& L" y* O+ m. M& o: M1 P" }
- }
复制代码
4 u2 F3 P! l+ o+ n8 e* l+ `
! f$ n0 M" c2 p- a% o7 C
5 r; h+ t* s4 @! B# w e7 {如果是首次上电的,可以看到没有调用standby函数,所以他可以向下进入while1,while1中的函数如下 - if(HAL_UART_GetState(&hcom_uart[COM1]) == HAL_UART_STATE_READY)
- Y; n! N9 j0 Y) m - {5 m9 G- J2 z" B7 t0 _
- if(HAL_UART_Receive(&hcom_uart[COM1], (uint8_t*)&buffer[idx], 1, HAL_MAX_DELAY) == HAL_OK)
/ M3 U" t8 K$ _% l0 y% c, n - {9 C/ i o6 U$ _* D
- if(buffer[idx] == '\n') // 假设\r\n'作为结束标志
, d. Z7 z' ~3 e3 v' o; H: |+ _ - {. `' ]2 Z. M9 l. F9 z
- buffer[idx] = 0; // 添加字符串结束符$ }% m& W# b( O; I2 c
- idx = 0; // 重置缓冲区索引
- U, T: f& l) o3 H. A6 v - if(ParseATCommandAndSetRTC(buffer))
, t, r ^ x# `6 c$ f5 |2 y' X - {* r- t N$ C$ H& Y3 H, [0 r9 Q- N
- printf("Parse AT Command success\r\n");
& y. o; [/ ]0 C! ?% _, [9 M: A - /* 刷新屏幕 */8 e b) S9 ^% |2 K4 l
- E_Paper_show_calendar();
1 r" L# K9 I J5 B: W - printf("enter standby\r\n");% F. p, Q9 g. s8 F/ l. q, n- d
- /* 进入standby模式 */
- ^! O, Z/ V3 P; j4 P - HAL_PWR_EnterSTANDBYMode();
% y4 r1 w& a( r: ] - }! o F4 h% b, D1 b: Q
- else
+ I8 l2 q8 a5 e0 t( V9 q" l+ y9 g - {
/ s8 U/ X% k0 T4 q - printf("Parse AT Command fail\r\n");* G: r; O2 @& F( z
- }
, Q. y; U$ C( i) F# T - }
! J8 ?6 f Q6 C% F - else6 ]4 v" O# m% K/ o, [4 n* T; d
- {0 |5 B# y2 Z- r6 j3 D B
- idx++; // 缓冲区索引增加8 v4 C K6 I) _: O/ `
- if(idx >= sizeof(buffer) - 1) idx = 0; // 防止缓冲区溢出
& }9 O9 ]: R: I- x { - }
/ Z' ~4 I3 o$ b0 A" Y - }) z+ a N, j( F& z& t
- }
( w- m: {( a3 V/ j9 s$ \
复制代码
! _- `# {" a7 c+ _5 Q, t9 D
6 B) S d( {2 T. L v% s& {) e 4 g: C/ l9 {6 z/ ]
& Z, t: j1 C! R: E9 q& P+ @会一直去获取串口收到的数据,如果找到\n就认为收到一包数据,然后去解析,解析成功就刷新屏幕显示日期,然后进去standby。反之继续接收
, n3 V+ O& Y& s8 o+ s1 V+ g下面看看解析函数 - bool ParseATCommandAndSetRTC(char* buffer)
0 ~5 a) f) r' b - {& v8 O0 T; g5 c) G: J
- RTC_TimeTypeDef sTime = {0};
5 Y/ U/ }+ Z& I) R* L - RTC_DateTypeDef sDate = {0};
' U+ A5 y; J8 e, k - unsigned int temp_year;
' k' a9 ~% C7 Y2 ] - unsigned int temp_month;
3 {! ]( _& B2 L# ]) ` - unsigned int temp_day;$ V7 Q) U) y" \) x' |6 A3 Y
- unsigned int temp_weekDay;) L7 H9 J+ B1 ?9 v7 r- q
- unsigned int temp_hour;
; C) b: J8 Q: e1 e& m: D2 k - unsigned int temp_min;9 l/ I. y* V/ w
- unsigned int temp_sec;
5 F# I# n) P* R. C# m3 g - ! r! M) A2 m2 Y$ y1 D G; ~& o
- /* 检查命令的头对不对 */4 p3 |$ d5 W+ c4 k* ]
- if(strncmp(buffer, "AT+configRTCdata=", 17) != 0) return false;& b( ^/ Z. {5 S4 v/ R
: u) L y' ^2 x! f% q$ `: c% e- /* 跳过命令头部分,再跳过年的前2个数字 */2 O2 z" W5 K7 P, W$ N( I+ Z& @
- char* dataPtr = buffer + 17 + 2;
# i/ W$ j0 f$ z" s
. K9 B- N9 Q3 [- /* 解析年、月、日... */; B2 b- n0 {- O& N: W
- if(sscanf(dataPtr, "%2u-%2u-%2u-%2u-%2u:%2u:%2u",
) q' b$ K+ o8 q" N* G - &temp_year, ( V7 x: c% W) e
- &temp_month, $ l { B+ Z. j0 Q9 T
- &temp_day,
; g$ j2 w9 o. O2 W! F - &temp_weekDay,
7 T( W( b4 }" p. V4 u - &temp_hour,
W% @2 o3 p3 c, a - &temp_min,
q t& t3 O9 |, u4 r* c. J - &temp_sec) != 7)
$ r7 m p+ ^) e6 |0 Q - {, V. l3 X9 _8 a: K( R& [4 V
- return false;
; B; n! n6 ^' s - }
- q. R: S J' {5 _# M - / q, s# w' r ^" s+ B- _. S1 W
" C$ ]. \! X7 h7 ]3 ~+ e- sDate.Year = uint_to_bcd(temp_year);
: S B& F1 S3 W* {) I" k# m i, i0 u+ m - switch (temp_month)8 u3 g1 A* Y9 [. W- i2 |
- {' n! W2 o* W/ Q- @1 p2 G
- case 1: sDate.Month = RTC_MONTH_JANUARY; break;
7 M7 q$ u" r2 \& y: z9 ` - case 2: sDate.Month = RTC_MONTH_FEBRUARY; break;
! K t2 h2 {# T( | - case 3: sDate.Month = RTC_MONTH_MARCH; break;* J0 [! `4 M* d1 D6 P
- case 4: sDate.Month = RTC_MONTH_APRIL; break;
% [- y* }( ~# u ~ - case 5: sDate.Month = RTC_MONTH_MAY; break;& u# h* b- M! W/ k
- case 6: sDate.Month = RTC_MONTH_JUNE; break;
2 o7 m, Q' I" V7 G' ?$ s - case 7: sDate.Month = RTC_MONTH_JULY; break;8 A* `+ N' f$ C* d2 r
- case 8: sDate.Month = RTC_MONTH_AUGUST; break;6 q; l. h+ Q, a
- case 9: sDate.Month = RTC_MONTH_SEPTEMBER; break;# \3 X9 b( v2 U3 J$ p! K; X4 M
- case 10: sDate.Month = RTC_MONTH_OCTOBER; break;
9 c5 D9 D) l% O& @ - case 11: sDate.Month = RTC_MONTH_NOVEMBER; break;9 o9 B1 R- H$ I( K! v
- case 12: sDate.Month = RTC_MONTH_DECEMBER; break;: S. z* U/ F1 P4 G0 {0 F2 G
- default: return false;
1 K/ j2 l N, t r! F% R - }( h, ?* \2 F8 Y% ?2 y2 v7 j
- sDate.Date = uint_to_bcd(temp_day);
+ I) f$ M/ L7 D, L) k - switch (temp_weekDay) c8 _) B2 A. B' H, u: j9 p |) E
- {3 ~- \. M4 G3 L+ C8 z! E( T! o& R
- case 1: sDate.WeekDay = RTC_WEEKDAY_MONDAY; break;6 S) l$ x5 q7 n; i+ y
- case 2: sDate.WeekDay = RTC_WEEKDAY_TUESDAY; break;
0 S& @2 {+ x( o0 a; o, d; ` - case 3: sDate.WeekDay = RTC_WEEKDAY_WEDNESDAY; break;
4 u3 a$ h& b- h0 t; V - case 4: sDate.WeekDay = RTC_WEEKDAY_THURSDAY; break;
8 b/ l# S j0 V/ ` o1 c9 [ - case 5: sDate.WeekDay = RTC_WEEKDAY_FRIDAY; break;
7 c u7 z. T2 ~3 u - case 6: sDate.WeekDay = RTC_WEEKDAY_SATURDAY; break;
( Z- h# F. l0 n8 J( K - case 7: sDate.WeekDay = RTC_WEEKDAY_SUNDAY; break;
3 J. J8 M% w- K4 P% E/ a ^ - default: return false;8 }3 O. V# o5 ]1 P
- }6 D( a0 [4 C2 y
3 Q) @3 a( \( V# Y ?- sTime.Hours = uint_to_bcd(temp_hour);
5 F, V9 o h, A2 E - sTime.Minutes = uint_to_bcd(temp_min);
1 m8 R+ e) d. ]* ~& W( i, [% J$ V - sTime.Seconds = uint_to_bcd(temp_sec);
+ ~5 x' Y) u( J! D' r - sTime.DayLightSaving = RTC_DAYLIGHTSAVING_NONE;
1 E, w4 l2 [# p2 b! |1 v F - sTime.StoreOperation = RTC_STOREOPERATION_RESET;5 H3 d, {+ ]" S1 q
' S" h% J- t# U- set_rtc_data(sTime, sDate);7 F, G& M. y) N: w% x" R+ u! ~0 Z
+ u% P' X2 Q1 K7 _; I0 }- print_rtc_data();
6 V) K: L) G1 h5 W2 V ~! V - : E9 }; n2 O6 z- C7 p8 \# p
- return true;
4 w" \" P* v% x7 X. z' B4 U - }( Z2 R. g( k* ?: ~4 z
复制代码
) D* X8 h- w! r( Y' K0 Z如果解析成功就设置RTC,反之返回false。 7 f) ]5 b3 P4 {% N
; w6 f5 a- f/ S* l. ?# x" C我输入的串口数据是十进制的,但是RTC使用的是BCD,所以需要转换一下,这就是uint_to_bcd函数,他的内容如下 - uint8_t uint_to_bcd(unsigned int num) . x( S" E) D) A* f D% b
- {, C6 a+ H( i, B, b" J
- // 检查输入范围,确保0-99之间1 U4 h8 P! F+ A+ a6 d
- if (num > 99)
6 v2 U0 A' O `7 y- Q1 } - {0 e0 Q A* o, {' |% C/ |
- // printf("Error: Input out of range. Maximum 99 allowed.\n");
, h; z8 L! A& d& `% j: j7 d - return 0;
2 L3 H: h0 ^( F4 D! U - }
2 C* c; M2 O R1 Q) D: c
/ f, G& m$ l f. F+ u- // 将十进制数转换为BCD
% E6 Z, H J% W% @) Y9 E8 y - uint8_t bcd = (num / 10) << 4 | (num % 10);! f" q" t6 T9 E
: t- ]4 k4 _' s& L- return bcd;
2 ~2 K, V$ X5 l% N3 n: Q. ~ - }
8 ^0 O2 B/ R5 @$ q( Z1 D V. h
复制代码
# V- S# w9 S1 Z2 T" f核心的逻辑代码都在这里了,屏幕显示的函数与具体逻辑无关,我就不展示了,可以自己去看附件中代码 : C& t5 _6 C- _
e: `" o2 ? Q, H9 h5 Y四、上位机注意事项 ; U! m/ m, V' Y+ G: J* J% c
上位机软件是用python+tkinter+serial写的,所以保证你的电脑上安装了python环境(我这边是最新的3.12.4),然后要安装tkinter、serial的库,否则运行时会报错:“ModuleNotFoundError: No module named ‘tkinter'”或“ModuleNotFoundError: No module named ‘serial'”(我忘记是serial还是pyserial,大致就是这么个意思)
6 u5 ^- I+ t% _$ ?/ }4 i& [% I: {& m; ]
" S; X( y- F+ d2 T: E0 k; G/ T
你可以用以下指令查看是否安装这两个库 如果安装了,会弹出一个窗口,这是tkinter的一个demo 如果安装了,会打印 出可以用的COM口
. F% f; d! R& ?5 |: F如果你没安装,可以用以下命令安装serial tkinter一般都是在安装python时自动安装的,如果你没有可以看一下这个 文章 $ u9 ^, v( M+ b3 X/ t( F! p
然后上位机我也提供源码在附件,如果EXE实在无法运行,可以尝试用VSCODE调试运行,我这边之前就是调试可以运行,EXE无法运行,需要安装东西,很奇怪 . I2 V% d; l/ n( i5 n
然后便于测试,我提供了强制写入23点59分30秒的版本,可以通过注释以下代码来实现 , v# k# A, V* A
# _+ G% d0 c! P U3 a, n9 [
. }" w$ m5 S4 v! z+ i" C5 _- A: p0 e W% H/ e! k F% ]
五、后续改进想法
[0 b/ j. G/ d% a, |这个demo现在其实处于一个初始状态,低功耗部分还没有调试过,目前已知的硬件上肯定需要做调整,否则功耗挺高的 * {- i+ W! k" \/ u
1.LED3这个灯要干掉,他一直亮着耗电
6 O9 w5 J3 }" k# D$ f+ s2.供电及串口,之前我测试standby功耗是是选择使用CHG跳帽的,现在为了可以借用STLINK的VCP被迫选择STLK,这会增大功耗,后续我打算使用外部的串口,供电改回CHG。板子上需要把SB48/45焊接一下,这样串口可以引到arduino的D1、D0口上,配置完时钟后拔掉接线 2 q& u6 ]! ]1 o4 a" e8 K
- _% r, ]0 v9 T( a( f
1 C0 w1 L/ V; W6 Z0 z j( H
: O5 ]) J# F% E; ?
3.STlink VCP的R23、24电阻干掉,我一直怀疑MCU会有电从这个漏过去,导致之前测量的功耗高,反正现在打算用外挂的串口,这个我就直接干掉。 % p6 ]" E) G$ v$ G2 @% K
4.配置RTC的机制有点不灵活,用户想要配置RTC,必须断电再上电,不友好。毕竟RTC时间长了就会偏,手动重校时是必须的。后续可以把板载的用户按键利用起来,把它作为唤醒源,当按下后,MCU WKUP,等待用于输入配置指令。多久没收到正确的指令就再进入standby
2 Z. d- u( ^5 Y6 w" z) G* ]5.美化一下界面
8 x; L6 W7 i; ?7 y1 `6.增加wifi模块,这样可以实现每天网络校时,并且还可以获取天气预报信息,让屏幕看上去不那么空
. F9 c: N& Z' c7.如果有合适的液晶段码屏,可以加上,用于显示温湿度数据,躲开了电子墨水屏的问题
/ u. o" H. H$ }- z$ }
2 s1 u4 N: x2 [) i+ [4 V5 x
, e0 M* Q' {0 C' r' E" s6 S- N/ s* I! d- `9 X; l' k
4 N* c2 F) H3 X感谢各位读到这里,如果你有更好的想法或者有疑问或者代码有错误,欢迎在评论区交流 4 V5 [3 S" y$ w7 q! S
D+ l8 R) [" C, {' T( t0 W! H7 l4 J& D
1 z A1 |' | J8 X3 ?8 |
# ?( y# m$ J, L. W
2 a4 i$ C- b* f5 N项目源码及上位机如下 0 }* j$ \. V; X* C
|