本帖最后由 1五湖四海1 于 2015-4-18 10:49 编辑 , a [8 I, T! H G
9 F1 r/ E7 m1 G6 _% m* j
我这小项目是为了解决运动的活塞是否被卡住而设计的。在实际中汽车刹车系统出现故障容易造成重大事故发生。我们的矿车有两个加力器,如果一个加力器出现问题能及时发现是可以避免事故的发生。所以制作这个运动部件卡住报警器对安全是很有必要的。 实现原理上用一个能够检测距离的模拟量传感器。控制器先判断运动部件是否受人为控制开始运动,如果部件开始运动,控制器的模数转换器每隔一段时间采集一次距离,控制器计算前后两次采集距离值是否一样,就可以判断运动部件是否被卡住不动了。下面介绍下用到的器件和具体设计细节。 距离传感器用的是光电模拟量输出传感器,检测范围在10mm——100mm。如下图中两个圆柱形的便是距离传感器。 控制器用的是ARM cortek M3 STM32f103,有人会问为什么要用这个高性能的处理器呢?我认为现在已经不是51单片机是世界了,51单片机没有AD转换器,需要外接AD转换器,代码需要从零开始写。而ARM cortek M3 里面有10bit AD转换器,有大量的写好的库函数可以调用,可以缩短开发时间。他们的价格差不多为什么不选STM32f10。这次用的是现成的STM32f103最小系统板。如上图 代码设计上用到了两个定时器中断,一个定时器每隔200ms采集一次距离值,另一个发生故障时每隔500ms报警一次,用到的算法有AD软件滤波,采集500次距离值后,用冒泡排序法去掉最大值和最小值后,再求平均数算出距离值,这时采集的距离值很准。剩下的就是用if else编写的逻辑判断了。调试时用到了printf()库函数,结合电脑串口调试助手调试代码。调试代码如下图,部分程序代码在最下面 完成的控制器如上图,正常工作时指示灯和蜂鸣器不亮也不响,如果活塞被卡住时声光报警。 上图是加力器,它是用气顶动油完成输出二倍力的器件。如果活塞卡死这个报警就会报警提示故障。 下面是代码是主函数代码 最下面有全部代码如果有问题可以和我交流,QQ:835358518 微信:hm15041303104
+ Y$ J; P6 {! H/ @) }# m6 x#include <stm32f10x_lib.h> #include "sys.h" #include "usart.h" #include "delay.h" #include "led.h" #include "key.h" #include "exti.h" #include "wdg.h" #include "timer.h" #include "adc.h" #include "alarm.h" ; a) @0 B K8 A% L
#define LIM 5 //卡住判断参数 #define TREAD 2 //踩刹车 #define LOOSEN 1 //松刹车 #define CLEAR 3 //清除 #define DIFFERENCE 50 //回差
* ?; E( U [2 O7 s/ w
" f& Q6 x& j4 O9 }8 tu16 lim_value = 1800; //极限报警值 u16 max_value_l = 1000, max_value_r = 1000; //踩刹车标志值 u16 min_value_l = 3000, min_value_r = 3000; //松刹车标志值 $ b# e, g! k& A6 [7 ]
u8 count = 0; //设置刹车标志防干扰计数 u8 lock_count; //卡住时计数防干扰处理 ! o9 T# f7 \" E: Y+ q
u16 distance_l[2] = {0,0}; //卡住比较数组 u16 distance_r[2] = {0,0}; u8 i = 0; //采集数据计数变量
) ~9 ^" ^4 O3 X1 m' I. obool brake_state_l, brake_state_r; //刹车状态标志位 bool lock_alarm_l, lock_alarm_r, lock_alarm; //卡住报警标志位 . J& V# l. ]) Q& V- `" Q
u16 value_buf[N]; //滤波求平均缓存 u32 sum; //滤波求平均总和缓存 u16 temp; //滤波求排序缓存
0 T8 \/ f! j1 T. L //判断是否活塞卡住 bool is_stuck(u16 distance_zero, u16 distance_one, bool brake_state, u16 *max_value, u16 *min_value); void data_collection(void); //数据采集 /** 主函数 */ int main(void) { bool lim_alarm_l, lim_alarm_r; //超限警报标志位
1 N9 M# P/ ]) k2 ]/ u4 U- h) D Stm32_Clock_Init(9); //系统时钟设置 delay_init(72); //延时初始化 uart_init(72,9600); //串口1初始化 LED_Init(); Adc_Init(); Timer2_Init(5000,7199); //定时报警 Timer3_Init(2000,7199); //定时采集ADC // Timer4_Init(5000,7199); //定时打印 1 j$ c0 F) q, n
while(1) { lim_alarm_l = isOverrun(distance_l[0]); //返回超限标志 lim_alarm_r = isOverrun(distance_r[0]); //返回刹车中标志 brake_state_l = isBrake_state(distance_l[0], max_value_l, min_value_l); brake_state_r = isBrake_state(distance_r[0], max_value_r, min_value_r); //报警处理函数 Alarm_dispose(lim_alarm_l, lim_alarm_r, lock_alarm); } } /** 定时器2中断服务程序 功能:发生定时中断控制蜂鸣器报警 */ void TIM2_IRQHandler(void) { - V6 H% H1 I7 O' K. b4 p- \ O4 i
if(TIM2->SR&0X0001) //溢出中断 { BUZZER = 0; delay_ms(100); BUZZER = 1; } TIM2->SR&=~(1<<0); //清除中断标志位 } /** 定时器3中断服务程序 功能:发生定时中断时采集ADC数据,并根据判断活塞是否卡住 */ void TIM3_IRQHandler(void) { if(TIM3->SR&0X0001)//溢出中断 { if( ( lock_alarm_l && brake_state_l) || ( lock_alarm_r && brake_state_r) ) lock_alarm = 1; else lock_alarm = 0;
& O2 J$ q) T r1 a' z' w0 v data_collection(); //数据采集后判断是否活塞卡住
' V8 l! W" H1 E) W } TIM3->SR&=~(1<<0); //清除中断标志位 } /** 功能:1.判断左活塞是否卡住 2.设置刹车标志 */ bool is_stuck(u16 distance_zero, u16 distance_one, bool brake_state, u16 *max_value, u16 *min_value) { u8 motion;
( d% T% e: L: x* i7 R6 \; E if( (distance_one-distance_zero) > 5) //松刹车标志 motion = LOOSEN;
W: J; \; P9 g( q8 y if( (distance_one-distance_zero) < -5) //踩刹车标志 motion = TREAD; 6 `' y7 r# J7 Q3 i: r7 h* g E
if( ( (distance_one-distance_zero) < LIM) && ( (distance_zero-distance_one) < LIM) ) { /* 踩刹车时动态设置最小值刹车标志*/ if(motion == TREAD) /* 如果这次值比上次设置的标志值小说明上次是卡住值,更新最小值标志*/ if(distance_zero+DIFFERENCE < *min_value ) *min_value = distance_zero+DIFFERENCE; ' p* r2 T2 U0 p, T4 J
/* 松刹车时动态设置最大值刹车标志*/ if(motion == LOOSEN) /* 如果这次值比上次设置的标志值大说明上次是卡住值,更新新的标志*/ if(distance_zero-DIFFERENCE > *max_value ) count++; //防干扰计数 if(count > 1) { *max_value = distance_zero-DIFFERENCE; count = 0; } ( }3 P# z4 s) T9 R; U, v) E
motion = CLEAR; //清除刹车标志
7 h. o$ O; x+ c* D" n if( brake_state ) //刹车时如果卡住开始计数 lock_count++;
" |! _5 @" H }: _8 n& ?; k0 S if( lock_count >= 5 ) //计数五次后报警位置位 { lock_count = 5; return TRUE; } } else { // lock_alarm = 0; lock_count = 0; return FALSE; } } /** 功能:判断活塞是否卡住数据采集函数 */ void data_collection(void) { / d+ R5 p6 C) c$ ~
u16 distance_zero, distance_one;
% q* Y5 Z+ a4 x5 E4 T filter(ADC_CH0); distance_l = sum; 6 ^0 F7 w. Q$ `3 q; N+ d( W: Y; h$ s
delay_ms(20);
0 G& _# B* M0 [# v6 C5 p {" V filter(ADC_CH1); distance_r = sum; i++; if(i > 1) //采集了两组数据后判断一次是否卡住 { i = 0; distance_zero = distance_l[0]; distance_one = distance_l[1]; lock_alarm_l = is_stuck(distance_zero, distance_one, brake_state_l, &max_value_l, &min_value_l); distance_zero = distance_r[0]; distance_one = distance_r[1]; lock_alarm_r = is_stuck(distance_zero, distance_one, brake_state_r, &max_value_r, &min_value_r); // d_value = distance_l[1]-distance_l[0]; // printf("%d,%d\n", motion_l, motion_r); printf("%d,%d,%d,%d\n",max_value_l,min_value_l,max_value_r,min_value_r); // printf("%d,%d,%d,%d\n",lock_alarm_l, lock_alarm_r ,lock_count_l, lock_count_r); // printf("%d,%d,%d,%d\n",distance_l[0],distance_l[1],distance_r[0],distance_r[1]); // printf("%d,%d\n",distance_l[1]-distance_l[0],distance_r[1]-distance_r[0]); // printf("%d\n",distance_r[1]-distance_r[0]); } }
1 m) b2 C4 t, O 8 K3 X1 a& ?2 |' l+ `# c3 I: K9 h3 ]
# H, F* T% g+ B _( B1 f
; |; X& Z3 r6 p% q; B! v
% t' d8 D5 h' r, n' n0 d* R+ Z1 [+ i4 p" _- C# l7 D `( V5 N
/ @! A5 v; j4 X* s1 D" K6 k: j* K1 T* Y3 f, W7 _
' U. r6 G8 f9 a
0 F, h; D; G: u" D$ n3 R" B$ q) p' P6 O- D2 C: L# Z
1 y6 C& ^# Y, }$ t3 p- W
& g7 V) x! N/ v6 E* c5 c( V$ E& _: b v- D# I
/ ? ]- S0 |+ `& T
3 i( z% D+ ~& d9 u4 E; d/ F+ S6 f0 d+ l9 J0 G6 |
% l1 x8 T8 {! ^7 b$ b% V! }+ p
* j/ U+ o9 [: e
' ? o( A5 h$ |( @
: `9 {4 h3 L% {* o* P# g! ]5 _ |