套件概述* m# H; q/ Q: v" C* _
STHS34PF80 是一款非冷却、工厂校准的红外运动和存在检测传感器,工作波长在 5 µm 至 20 µm 之间。STHS34PF80 传感器设计用于测量视野内物体发出的红外辐射量。该信息由 ASIC 进行数字处理,可以对其进行编程以监控运动、存在或过热状况。. c, C& u0 H+ B: {2 p! F
! r2 ~9 b5 k! ~ d3 p" m
本章主要驱动STHS34PF80,进行人体检测。
. J* O3 q8 V1 ?: I: T. b, Y8 k! W. L( K! h( z
2 Q( \' J9 I2 [1 Y; N* U, G& v8 @2 u0 x/ x! N
7 [6 O6 U% ~- x3 ?& E& w
引导 # p' {$ R9 G/ A$ Z+ t4 h
在设备上电后,它会执行一个2.5毫秒的引导过程,将校准系数从嵌入式闪存下载到内部寄存器,并加载AVG_TRIM(10h)、CTRL0(17h)和SENS_DATA(1Dh)寄存器的默认内容。在引导时间内,设备的寄存器不可访问。4 p: r6 K4 J0 U( S. d; ^% ?
/ A4 C- V/ V$ a6 H: ?7 N+ h引导完成后,设备会自动配置为电源关闭模式。' Z0 @0 V, U" |4 W
/ ?% j( k7 y0 ]8 N; w在上电后,可以通过将CTRL2(21h)寄存器的BOOT位设置为1,重新启动引导过程,以重新加载上述寄存器的默认值。不需要切换设备电源线。. U; J- }3 F; W
7 }! ]' K' A( t0 r, k& s/ B0 E在重新启动时间内,设备的寄存器不可访问。重新启动完成后,设备会自动配置为之前的操作模式,并且BOOT位会自动清除为0。8 h8 o9 }4 ^8 r I
, L. |+ L3 T+ S) h6 l; b
& k/ }) n6 h2 Q9 A- g4 C# ~/ X
( a/ `4 w" m, A6 u5 v0 p# [
+ |4 J$ V; [. ~; b; I温度测量滤波方式
+ h( h/ @$ |# C/ w0 f首先需要设置AVG_TRIM (10h) 寄存器用于配置温度平均值滤波的参数,以控制环境温度和目标温度测量的平滑度和稳定性。在温度测量中,可能会受到噪声和干扰的影响,这可能导致短期内测量值的波动。通过应用平均值滤波,可以平滑这些波动,从而得到更加稳定的温度数据。AVG_TRIM 寄存器中的设置会决定平均值滤波的级别,从而影响滤波的时间常数以及平滑度。
' ]6 C8 B! M4 W! d1 U6 S5 R# g( o% a! _
这里默认参数如下2 @$ Y7 `# U. H/ J6 n& n
在AVG_TRIM(10h)寄存器中写入02h // AVG_T = 8,AVG_TMOS = 32
! q8 C/ K; A2 `% M在CTRL1(20h)寄存器中写入07h // ODR = 15 Hz7 C5 v5 x$ a( }( l2 e3 Q
H' Y; l2 u8 z* _& B0 ?, {
[; U* ^8 R# o. C. e8 C' o* s% r) E# r2 q8 g( s1 M
2 z0 J) B' c3 g' h4 P* B! M
AVG_T[1:0]: 这个设置位用于选择环境温度的平均值滤波样本数。
! }6 [, j( R hAVG_TMOS[2:0]: 这个设置位用于选择目标温度的平均值滤波样本数以及与之相关的噪声水平。4 z# k' l9 |; f; b$ ~2 r+ o
3 J. n8 R% d5 U8 F+ A+ X
+ q _' G; [ D% H( Q+ }( u/ F, l6 |
, C9 e6 I, j$ k* v
' @/ Y, C2 H, i: h) C8 q
参考demo中将AVG_TMOS [2:0] 和AVG_T[1:0] 分为了2个函数进行操作。8 C9 b$ p- c, s' R
5 n( N' f2 ?9 ~! b4 X9 P: S
, j: F. _( }0 N2 {& m$ O. P6 s
) c9 K& I9 ]* v+ P
0 r: _; J9 ]' A. @' A; j
7 a2 ^6 ]1 `' Z% L4 ]+ {; X7 g修改后操作如下所示。
" y% C2 m7 q- s1 p7 F4 D- M- /**
5 f6 F' e; e$ ?) P6 Q - * @brief Select number of averages for object temperature.[set]
1 R- c! G' x7 J+ d - *
: G3 Z+ H4 p& s/ D/ y - * @param ctx read / write interface definitions
q0 F. o, L( Y - * @param val AVG_TMOS_2, AVG_TMOS_8, AVG_TMOS_32, AVG_TMOS_128, AVG_TMOS_256, AVG_TMOS_512, AVG_TMOS_1024, AVG_TMOS_2048,
. F& x& P1 O0 j4 v; ?6 ^ - * @retval interface status (MANDATORY: return 0 -> no Error)% Z; D# F7 \7 _# T
- *
% _8 ^* L4 {7 O0 a! X F6 O0 K5 B - */
0 B: w3 r5 i" u - uint8_t sths34pf80_avg_tobject_num_set(uint8_t add, sths34pf80_avg_tobject_num_t val)
7 Q' Y ^2 ^! q3 M: a+ n' b - {
. ` C$ l, G D! i% r - sths34pf80_avg_trim_t avg_trim;
0 t0 v$ {+ z" E6 [. v9 @ - int32_t ret;/ Y" f$ a* D) M6 L F
- ret = sths34pf80_read_reg(add, STHS34PF80_AVG_TRIM, (uint8_t *)&avg_trim, 1);4 {1 X$ ^) K- z- S9 b. v9 e0 X
3 ~ _8 c8 V( [! K7 B+ ^- if (ret == HAL_OK)
* M. `! x6 ^9 S$ E" t2 ^9 q G - {
. q; \9 U% x9 Z) e - avg_trim.avg_tmos = ((uint8_t)val & 0x7U);
+ y T+ M' k% C* s- E2 A - ret = sths34pf80_write_reg(add, STHS34PF80_AVG_TRIM, (uint8_t *)&avg_trim, 1);# N' E4 m r' w
- }5 \$ F+ P* ^/ @' v. |
- / G2 ~4 }7 B* f( e5 J
- return ret;2 ?# C I& W( }4 P r, ?
- }
' j+ }/ ?) C+ q* \ - /**
8 D: M: Z) t6 c7 ] - * @brief Select number of averages for ambient temperature.[set]
: W( I" Z( [+ f6 ~ - *
( W- H t+ h1 b" @9 f - * @param ctx read / write interface definitions
7 {( v _0 I8 s8 V3 C8 R - * @param val AVG_T_8, AVG_T_4, AVG_T_2, AVG_T_1,. A' N; m* J% t; x3 a
- * @retval interface status (MANDATORY: return 0 -> no Error)
$ M5 d5 i. M2 z# X- r - *3 t4 E# M; F2 D. R( ^7 x
- */
# x" F# h/ M% n8 U' n - uint8_t sths34pf80_avg_tambient_num_set(uint8_t add, sths34pf80_avg_tambient_num_t val)
, J" q q: g8 S( e; f - {+ `! C! s8 G+ ~, H
- sths34pf80_avg_trim_t avg_trim;) F6 j2 Q7 h( Y1 s: R8 t: r
- int32_t ret;
6 G3 k) i; [& Y' A8 |. k z1 I - ret = sths34pf80_read_reg(add, STHS34PF80_AVG_TRIM, (uint8_t *)&avg_trim, 1);
^; X7 k% u9 Q. z0 v7 Z8 @ - if (ret == HAL_OK); ^. n) E& a1 y4 ?) V
- {
/ x3 w! a# h R$ O - avg_trim.avg_t = ((uint8_t)val & 0x3U);
( [) ^7 \% ^8 `9 i, }* \) Q2 t - ret = sths34pf80_write_reg(add, STHS34PF80_AVG_TRIM, (uint8_t *)&avg_trim, 1);
6 W- T5 t& K3 i1 n& D - } " A* u- c* d% f: q
7 h1 J5 R2 W( ^+ {+ p) G9 o1 r7 L- 7 w6 o( ?- f `) ?# s
- return ret;
- p1 k7 Z% i3 Z$ A9 k# }! S - }
+ d% M' v! h# {: I5 v; v
复制代码
( {5 a1 C4 c% x' _8 k1 f使用如下所示,这里设置参数为STHS34PF80_AVG_TMOS_32和STHS34PF80_AVG_T_8。- /* Set averages (AVG_TAMB = 8, AVG_TMOS = 32) */
% x+ Z0 T" ~" u - sths34pf80_avg_tobject_num_set(STHS34PF80_ADDRESS, STHS34PF80_AVG_TMOS_32);# o2 {" Y* _) _
- sths34pf80_avg_tambient_num_set(STHS34PF80_ADDRESS, STHS34PF80_AVG_T_8);
复制代码
9 o1 {4 e, L7 s! k智能识别算法
# ?6 W7 M9 l; J2 JSTHS34PF80嵌入了智能数字算法,以支持以下三种检测模式:4 }" Q6 ~: v# Y" V
• 存在检测
2 l& I) j& p: V7 r1 w, N$ O- `4 g$ ]& E• 运动检测" K' @8 M/ I5 A. }$ ~$ u+ q
• 环境温度冲击检测
: |% H' ]) Q( U Z" Q! ]+ t# R1 I$ W! ], L5 y
这些算法分别使用不同的低通滤波器(LPF_P、LPF_M和LPF_A_T)。此外,存在和运动检测算法使用另一个共同的低通滤波器(LPF_P_M)。这些滤波器用于生成中间信号(TPRESENCE、TMOTION和TAMB_SHOCK),可以用于对算法本身进行微调。这些滤波器的截止频率值可以通过它们各自的位范围进行配置,这些位范围可以在LPF1(0Ch)和LPF2(0Dh)寄存器中找到,如下所示。
/ S) T _: u5 a+ u- }2 E( ~4 {
1 R" R+ G' c- \% E
3 v2 u2 H9 p4 S; t: P4 r& T
& d1 |! ]: u$ x* a$ M寄存器LPF1 (0Ch)和LPF2 (0Dh)如下所示。
# B' o$ j# [. a \: v" ]2 L+ Y* r0 G* e1 f
+ t* C3 [! S& Q9 h- a+ Y! ]1 B W* E8 Q" y
案例中只是对滤波器进行了读取以及打印。
4 [3 G. F, Q$ ~7 V; j2 _: ~6 N. s- f6 S" A" i" J2 X/ K/ H# [
# X' G9 {- e3 w3 H G
( ^5 g" S: G8 R* |( `! R" @* B4 v修改后如下所示。
* H: D# @. Q3 T! s6 l1 {9 F7 D' A使用块数据更新(BDU)功能 / j" ?# `. L% h* N2 s$ s
如果读取输出数据的时机与数据准备信号不同步,强烈建议在CTRL1(20h)寄存器中将BDU(块数据更新)位设置为1。& U% r# j* c/ g8 ~, U* q
4 v7 K' {* F- [' F5 e0 b. m2 J/ r7 f
此功能可以避免读取与不同采样相关的寄存器值(输出数据的最高和最低字节)。
+ ? l, M4 B ~6 F0 \ G7 X; ~4 N5 ?4 e8 k+ t. g$ i
4 u* f/ ~7 Y4 K2 _" W$ j6 B+ B; p
. j: u& [ h" J$ \- z; Z案例也是设置了BDU。
) \/ \* \6 ?3 F7 x( D' ]+ ~
- N! \ `8 `* X+ N
3 S" X9 N. H& U( }
% l% C3 u4 u. B% J( ^2 f( dBDU在CTRL1 (20h)寄存器中。2 S [( `) W' r
& O; `; _+ }3 i
& d8 i$ ]2 Y7 r5 ]! v* w( s2 D1 ?
; p; D" X% M1 X6 u, n2 J( t修改后的操作如下所示。- /* Set BDU */
% o, @6 E( O" e& j$ K/ { - sths34pf80_block_data_update_set(STHS34PF80_ADDRESS, 1);
复制代码
2 P1 T! G0 N/ a2 a/ Q* y7 P设置ODR速率 + ^* z2 @& x0 _( D. S' e8 c4 N
初始化完毕之后需要通过CTRL1(20h)寄存器中的ODR[3:0]位范围选择其中一种操作模式(连续模式),或者在CTRL2(21h)寄存器中将ONE_SHOT位设置为1(单次触发模式)。% x* L7 e; V1 q- W' Y
& z7 k i9 f1 j6 b# u9 o, }
3 c R% `& _; g3 \/ O9 ]4 j
- A2 I9 Y5 i/ V! u; k4 W当配置了寄存器LPF1 (0Ch)和LPF2 (0Dh) 的LPF_P、LPF_M和LPF_A_T、LPF_P_M滤波器之后,可以去设置ODR速率。) ?% [7 f. s' F- o* C( J
' G% Q5 P5 E8 m5 n! _$ `
7 [ M7 Z! |- u E6 z/ \1 A7 ]0 L
$ Q1 k9 U! h5 T3 O1 u6 G- M这里设置速率为30HZ。
6 y* I& g, I' M3 K; o- /* Set ODR */
8 ]4 w! F) v# K$ ]" R! g% C r4 y' C - sths34pf80_tmos_odr_set(STHS34PF80_ADDRESS, STHS34PF80_TMOS_ODR_AT_30Hz);
复制代码
|# z/ p% s9 y3 s状态寄存器
K$ ~3 A$ O8 G4 {# |7 j, [ I' n初始化完毕之后,可以通过状态寄存器STATUS (23h)获取DRDY标志位。数据就绪信号由STATUS寄存器的DRDY位表示。当新的数据可用时,此位被设置为1,并在读取FUNC_STATUS(25h)寄存器后被清除。
0 `1 g1 j' r9 w" ~4 t3 [- @8 w$ h0 B4 p+ l& Q& O
. g8 n @- m" _% O3 Y, w% y6 o- I P0 W+ Q. W
主程序中会去获取改状态位,若为1,则代表数据已经准备完毕,可以进行获取和打印。6 k3 `# r* E2 ^; Q* C* G! K
7 |/ t# v' H" `7 `% h6 Q m+ v8 S
- u5 ?) F8 \6 J- E4 S. w8 @, U6 R$ w/ Q+ |' E# ^: [- R
FUNC_STATUS(25h)主要检测三个标志位:PRES_FLAG、MOT_FLAG 和 TAMB_SHOCK_FLAG,这些标志位用于检测不同类型的事件。7 r: m: I1 }: i" o7 R/ F" p1 s0 Z
u) i) {1 t3 @* a; F
PRES_FLAG是存在检测标志位。当存在检测到人员存在时,此位变为1。
/ X# m; Q+ V8 @0 N% b% M2 U) R8 }/ I* ~7 j& v9 |% }
MOT_FLAG:运动检测标志位。当检测到运动时,此位变为1。
& n7 p$ L$ X. p7 K' V
- j1 x9 c3 T2 [- A1 ~; C( {TAMB_SHOCK_FLAG:环境温度冲击检测标志位。当检测到环境温度冲击时,此位变为1。
f8 C; |; I7 S0 x: G) N* s
U9 Q/ x1 l; B! p+ X& L2 P7 y
% c8 y) @8 g3 P& G$ ~& A
5 x* }+ X$ S" y* k9 Z修改完毕主程序如下所示。- /* Infinite loop */; d) o6 J$ v0 w e3 e1 Y1 _
- /* USER CODE BEGIN WHILE */
, G' a" [" R( x' v5 {% a. O. E - while (1)
: D0 `; k8 o/ w4 p" o# v - {
1 N g$ E+ {+ ^+ E3 ]* y/ s8 Z8 b& l - sths34pf80_tmos_drdy_status_get(STHS34PF80_ADDRESS, &status);
9 F7 V, T u% }, i - if (status.drdy)1 e. ^; ~* \0 b1 \1 X* @
- {
1 @/ W; U0 F2 k8 c3 ]' ~1 O# b - sths34pf80_tmos_func_status_get(STHS34PF80_ADDRESS, &func_status);+ p" Q: S# I; q. q2 O! u* T
- printf("-->环境温度冲击检测标志位 %d - 存在检测标志位 %d - 运动检测标志位 %d\r\n",func_status.tamb_shock_flag, func_status.pres_flag, func_status.mot_flag);( |' e/ T6 e% |6 k9 w
- }
V, h3 S1 e0 W+ b: i# v - HAL_Delay(1000);
8 Z9 _* q* ]1 y7 \8 {
5 a" t+ V6 Y9 G2 |& G- /* USER CODE END WHILE */
6 q2 x& N+ A$ J; i- I/ n! Q) a
8 _0 V0 |: G6 S$ i2 f3 B7 p* _- /* USER CODE BEGIN 3 */3 b. }6 X: a9 f% Y
- }% R& H# B3 p6 l5 j
- /* USER CODE END 3 */. g6 W6 @, Y; f4 C' j+ G
复制代码 ( v8 A9 [( f& u" u- a( t( i: d
main.c * h( k' u. U* Q- B0 ^6 |5 w
- /* USER CODE BEGIN 2 */' F* H$ L# U X- s
- sths34pf80_lpf_bandwidth_t lpf_m, lpf_p, lpf_p_m, lpf_a_t;
) u/ k' X( F& O% ^' w - sths34pf80_tmos_drdy_status_t status; ) S- m) a" f9 H( X+ ^# ^. c; B
- sths34pf80_tmos_func_status_t func_status;
4 _! o/ D$ C: C7 c( K- q! S - 0 x! ^" Z* u+ Q4 |0 y4 `
-
% K( _ r+ z7 [3 V: P4 b; _ - . S& r2 ?! Q- u4 t$ }
-
) e4 z- c1 U- L; m+ p+ A - HAL_Delay(200);
. x, H4 i& R& j - printf("123");" A) q2 f/ y$ e; }3 K
- uint8_t STHS34PF80_ID =STHS34PF80_getChipID(STHS34PF80_ADDRESS);# f6 D' i4 w0 F" q& `
- printf("STHS34PF80_ID=0x%x\n",STHS34PF80_ID);
0 r* h) n* p) b - if (STHS34PF80_ID != 0xD3)
0 w) q" ]5 X/ }+ z4 d2 |7 O' p - while(1);! H2 ~( B' G$ E" X- y- n5 g, D0 Q, Z
- /* Set averages (AVG_TAMB = 8, AVG_TMOS = 32) */
! c$ E, L9 n* h* g+ M. W6 j - sths34pf80_avg_tobject_num_set(STHS34PF80_ADDRESS, STHS34PF80_AVG_TMOS_2048);. Q, A5 V$ X/ b1 {- w$ p! n
- sths34pf80_avg_tambient_num_set(STHS34PF80_ADDRESS, STHS34PF80_AVG_T_8);0 W& P _. T$ ?# V4 h; c
+ {* K7 J! b( A- /* read filters */7 A& r# ]1 U* L* Z- o* ?; w3 r
- sths34pf80_lpf_m_bandwidth_get(STHS34PF80_ADDRESS, &lpf_m);
+ k+ w6 I! x; x3 t. E( W - sths34pf80_lpf_p_bandwidth_get(STHS34PF80_ADDRESS, &lpf_p);
" [$ n& ^/ k; f0 |3 Q1 }0 ~) l6 n: { - sths34pf80_lpf_p_m_bandwidth_get(STHS34PF80_ADDRESS, &lpf_p_m);9 Z4 @ A. j+ E6 V* ?$ m1 R5 f; j! V
- sths34pf80_lpf_a_t_bandwidth_get(STHS34PF80_ADDRESS, &lpf_a_t);
) J3 e3 A8 y, K6 b$ P; @ - ) T# z, I# L% B. Y6 g. A5 E$ L
- printf("lpf_m: %02d, lpf_p: %02d, lpf_p_m: %02d, lpf_a_t: %02d\r\n", lpf_m, lpf_p, lpf_p_m, lpf_a_t);
6 d Y7 x6 f; f" Q: p& { - 8 d( ~( z. I* e0 T6 U$ n- |/ E9 e
- /* Set BDU */
4 }% g6 @1 y/ j" D2 e8 R k& X - sths34pf80_block_data_update_set(STHS34PF80_ADDRESS, 1);) X: D) |, Q, h) W1 N
- / L3 ~9 d! f1 C9 |8 X
- /* Set ODR */
0 z/ A; M) S U9 B! ^# Y y5 O - sths34pf80_tmos_odr_set(STHS34PF80_ADDRESS, STHS34PF80_TMOS_ODR_AT_1Hz);
5 S: V5 E5 t+ U O+ P R9 Q6 f - # ?1 ^5 u! R! d% I
- int32_t cnt = 0;
$ Y. J0 g4 L! F: ^2 @+ n# d; S6 u. x -
: g. i9 {# ^- k6 q) ~6 g - /* USER CODE END 2 */
& d' G* O2 t+ q. ?* \ - , ]( O% C8 Z& P2 ^* }& D
- /* Infinite loop */7 F( |4 B. k: T8 s6 U+ z! C- q
- /* USER CODE BEGIN WHILE */% o8 Y- @& u- e7 p5 {$ @ X
- while (1)
! `8 c& B2 {% `1 p+ _ - {
" q0 [$ q4 b& n - sths34pf80_tmos_drdy_status_get(STHS34PF80_ADDRESS, &status);
8 \. P+ A& ]1 P. y - if (status.drdy)' B- s9 P2 }8 q v/ u1 I0 v9 T
- {; _1 Z" S) Z1 O5 P/ ?$ s; G
- sths34pf80_tmos_func_status_get(STHS34PF80_ADDRESS, &func_status);3 m/ C% y/ C: _7 f+ x5 e
- printf("-->环境温度冲击检测标志位 %d - 存在检测标志位 %d - 运动检测标志位 %d\r\n",func_status.tamb_shock_flag, func_status.pres_flag, func_status.mot_flag);: u. G) Q5 G) Y/ v
- }
: \/ y7 p) ?8 j - HAL_Delay(1000);
/ ?: u7 { f3 r j4 M6 N8 X
7 Q2 u9 L9 O$ n- /* USER CODE END WHILE */7 w" P: D0 K" m6 k5 k1 `
; u! u' h6 P5 @' Q- J- /* USER CODE BEGIN 3 */
* i' ?& p0 K& g - }+ P9 c1 b: D# |
- /* USER CODE END 3 */* T$ K5 m8 \+ n0 |. E
复制代码 5 {% I# c5 P1 q C* _+ G
转载自:记帖
& A$ _' J( Y" c( \如有侵权请联系删除
0 ]/ Q8 C+ x$ `# W* @- o# x$ I* a. A, B' F5 S! u e9 I
2 x+ u+ Z! B/ A6 s6 ^% y9 B% u
: Y2 W$ K" I% t+ c# ]: {) \0 c |