基于STM32温控系统的设计
$ J$ u7 k# W, n7 z5 ~4 k; q1 @ g; w3 v/ P8 U1 Q. `5 ~
一、系统功能描述 利用STM32系列芯片,作为本温控系统的主控芯片。同过RS232通信协议实现下位机与计算机的通信,通过XTR105和PT100铂热电阻组合转化为稳定的模拟信号量,然后通过AD7705将模拟量转变为数字量输入主控芯片,这样主控芯片通过计算就可以得到当前温度的值。然后,主控芯片经过PID算法,计算出需要输出电流大小,写到MAX538芯片中,该芯片将数字量转化为模拟量输入XTR110,从而得到稳定的电流输出。 ( K7 k0 X Q, P; | B4 d
! u! {% A) J' U1 J. K' o) J
# Q P; D" B- X6 w" t H6 m ^! G$ n二、硬件电路设计 $ V: l! C* V2 A- F$ k
1、硬件电路设计包括PT100铂电阻的信号调理电路、AD7705的使用、MAX538的使用、XTR110的使用。其中AD、DA的使用按照芯片说明书, 连接电路,并配置主控芯片即可。这里仅仅阐述测温的信号调理电路。
2 r9 T- ]. P# @! U# A" R' `: y根据TI提供的芯片的使用信息,这里实际的测温范围为0-500度,计算出上述电路中的电阻。
2 j; C( M1 Z' g% n" h9 [5 Z) c( y2、USB转串口电路设计 . f* [; p z: L
选用该芯片的好处:开发电路成熟,开发速度快,尤其是有配套驱动,基本不需在开发对应的驱动程序。
0 [+ D, S0 i% g9 a$ U9 G P# Q8 R1 a! i$ i
三、软件层的设计
" T4 h& O6 Z1 z1、 上位机软件开发 需求分析:上位机需要先设置串口通讯的基本参数,然后打开串口,设置温控的参数,设置完成后点击开始温控,最后接受下位机发出来的串口信息。 " {$ _, {: U1 Q% e: o& Y E
2、 下位机软件开发 需求分析:下位机需要完成与上位机的通讯,同时当开始温控时还要完成温控功能。下位机采用uC/OSII嵌入式实时操作系统,建立两个任务。
* m% t$ l2 _8 V- I J# E串口通讯任务通过任务解析模块对上位机发出来的指令进行解析,启动温控任务。温控任务将测量出来的温度通过Message的方式把内部的数据发送给串口通讯任务。串口通讯任务在将这些任务发送给上位机。 任务解析模块需要处理的事件有,串口参数设置、温控任务启动、温控任务结束。 : A0 g- E2 S( ?! A: g, {
3、 PID温度控制算法 Kp可以说是PID算法中必不可少的一个量。没有它控制就不可能实现。它的作用是根据当前量与设定量的差值按照一定比例放大后得到控制量。 - //pid.h
* Z# @" z, n7 W7 v9 C- P7 G/ g - #ifndef __PID__2 P Z/ `4 X& V( Q6 l& t3 `
- #define __PID__# N* c) n6 c7 a# Y* e1 F1 L4 I) h9 ^
- /*PID = Uk +KP*[E(k)-E(k-1)]+KI*E(k)+KD*[E(k)-2E(k-1)+E(k-2)];(增量型PID算式)
复制代码函数入口: RK(设定值),CK(实际值),KP,KI,KD 函数出口: U(K)*/ - typedef struct PIDValue
& |8 [% ~4 r6 I& X# | - {
& M# K* D) Y( w3 H - int8 KP;
; k9 q1 n6 T, y7 S - int8 KI;
9 l* d6 y; k7 @" c - int8 KD;- y5 S5 P% @8 Q3 d7 w C
- int8 F;/ H3 Z, w% J# _2 R6 c& P$ W0 {
- int8 BITMOV;
. v5 k% D( h* l7 H( |8 Q$ {! ` - int EK[3];
% O/ I. k9 W3 j, s5 j5 W0 [ - 6 V( h m; |! z0 [) O% \
- int UK; A3 X0 t/ x$ }" U9 `# S
- int RK;
& V# \) P- h0 W9 @, E - int CK;
9 u% x+ W$ {5 k6 g( Z* N1 t( I - int UK_REAL;
0 {' i9 B0 e, p8 \/ i9 y! z - $ `: M1 M9 A9 F6 q- l) P. Z
- }pid_str;% j. H: O" L4 q- |+ B
- //PIDValueStr PID;
+ [$ G& @. t* l7 k' s3 L - void pid_exe(pid_str*PID) ;8 |0 C$ z O0 m3 c
- #endif
6 n+ q# U7 K1 N6 i) A# B% z -
0 ^8 q5 U, r! |; E3 p' h - //pid.c9 k& r; ^9 n" Z
- /*PID = PID->UK_REAL +PID->KP*[E(k)-E(k-1)]+PID->KI*E(k)+PID->KD*[E(k)-2E(k-1)+E(k-2)];(增量型PID算式)
+ D+ F% ?' W, J& K5 @ - 函数入口:PID->RK(设定值),PID->CK(实际值),PID->KP,PID->KI,PID->KD- U i5 w2 O. ] v/ J5 _1 D& ^
- 函数出口: U(K)*/
3 {% ?9 \$ L! h7 A3 Q: w( j - #include"defines.h"7 N" c# ]3 q+ s! K% d Q4 I
- #include"pid.h"
0 c3 {% N9 G5 O" W - #define MAXOUT 0xff
* e. f& ^% s1 x; z - //#define MAXGAP 1002 y% [: a% D# {! U! m) D9 ^( ^
-
" n& X, Y7 w4 h5 L. x - void pid_exe(pid_str*PID)
+ X) s7 G& @% M - {
" C' S9 I! k% _9 Y9 _- |4 v J - PID->EK[2]=PID->EK[1];7 {4 |% K1 B' @& t, f
- PID->EK[1]=PID->EK[0];
3 y8 C3 p: F$ G x - PID->EK[0]=PID->RK-PID->CK;
9 z2 d6 s* \( X8 F8 @- R7 M - PID->UK_REAL=PID->UK_REAL
* y; G2 F1 h! ]& _% L& P# w4 v - +PID->KP*(PID->EK[0]-PID->EK[1])//微分一次后积分即原数
0 ^5 p5 x+ a" W - +(float)PID->KI*PID->EK[0]/PID->F//直接积分- H9 j& L) v( K
- +(float)PID->KD*(PID->EK[0]-2*PID->EK[1]+PID->EK[2])*PID->F;//二阶微分后积分即一阶微分; B/ a% ^+ X5 ~2 r) e# i" H9 ^+ X
- if((PID->UK_REAL>>PID->BITMOV)>=MAXOUT)
; @: `' _5 s. x4 N - {% |5 j+ o0 {0 c, c" z
- PID->UK=MAXOUT;! J6 Y4 g! Y' D+ D, H7 W+ |9 u) r
- }elseif(PID->UK_REAL>>PID->BITMOV<=-MAXOUT)9 i( L: I) y' Y) b7 d
- {
: ?4 _* ` X8 \2 U# Y - PID->UK=-MAXOUT;0 K `2 O5 ^* K0 l
- }else! F- u+ c) V3 n* X- h5 r
- {
* ]0 f' i. C# i/ Q! i7 f - PID->UK=PID->UK_REAL>>PID->BITMOV;
7 I; E9 ?6 ?/ R9 I& b - }
; A5 }; B3 p, F% e7 Y2 f - 0 r" i# U8 ~, v$ P- n- ~
- }
复制代码这里我写的代码用到的是增量型的PID(即UK_REAL +PID->KP*[E(k)-E(k-1)]+PID->KI*E(k)+PID->KD*[E(k)-2E(k-1)+E(k-2)];这句话所对应的是pid控制量在之前pid控制量的基础上增加的值,相当于求了一次导)。最终输出的结果将每一次运算的值累加输出就行了。 % v$ a! g5 q: a/ u3 l: C) }
|