我在做基于舵機調整方向的超聲波避障的藍牙遙控智能小車,用的是stc12c5a60s2單片機,超聲波用外部中斷和定時器1來測距,舵機用定時器0調整方向,小車的輪子轉動是用單片機的內置兩路PWM輸出的,單獨運行超聲波時不會影響到PWM的輸出,而運行舵機的時候就會導致單片機復位,求大神幫我看看原因啊!
#include<at89x51.h>
#include<STC12C5A60S2_PWM.h>
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
/***自定義一些數據*/
#define leftdata 0x11
#define rightdata 0x22
#define forwarddata 0x33
#define backdata 0x44
#define stopdata 0x55
/*****接線定義******/
sbit IN1=P1^5;
sbit IN2=P1^6;
sbit IN3=P1^1;
sbit IN4=P1^2;
sbit EN1=P1^3;
sbit EN2=P1^4;
/*****獨立波特率發(fā)生器專用寄存器*******/
sfr AUXR=0x8e;//AUXR的SFR地址在0x8e
sfr BRT=0x9c;//BRT的SFR地址在0x9C
//*********************************************************************************************************
sbit trig = P3^2;
sbit echo = P3^3;
bit flagg; //中斷進入標志
#define ControlPort P1_0
unsigned char push_val_left = 11;
unsigned char TimeOutCounter=0;
float dis=0;
unsigned char posit=0;
unsigned long S=0;
unsigned char const discode[] ={ 0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xBF,0xff};
unsigned char const positon[3]={ 0xbf,0xdf,0xef};
unsigned char disbuff[4] ={ 0,0,0,0,};
void Display(void) //掃描數碼管
{
if(posit==0)
{P0=(discode[disbuff[posit]])&0x7f;}
else
{P0=discode[disbuff[posit]];}
P2=positon[posit];
if(++posit>=3)
posit=0;
}
void delay_20us()
{
uchar bt ;
for(bt = 0;bt<10;bt++);
}
//*************************************************************************************************
void Forward(unsigned char Speed_Right,unsigned char Speed_Left); //申明一個“前進函數”
void Stop(void);//申明一個“停止”函數
unsigned char SBUF_DATA,flag;
//void SendString(unsigned char *pt);
void Send_Char(unsigned char DATA);
//L 左
//R 右
void Forward(unsigned char Speed_Right,unsigned char Speed_Left)//Speed_Right,Speed_Left為左右電機對應的速度參數0-255之間,255最快,0最慢。
{
IN1=0;
IN2=1;
IN3=0;
IN4=1;
PWM_Set(255-Speed_Right,255-Speed_Left);
}
void Back(unsigned char Speed_Right,unsigned char Speed_Left)//Speed_Right,Speed_Left為左右電機對應的速度參數0-255之間,255最快,0最慢。
{
IN1=1;
IN2=0;
IN3=1;
IN4=0;
PWM_Set(255-Speed_Right,255-Speed_Left);
}
void Turn_Left(unsigned char Speed_Right,unsigned char Speed_Left)
{
IN1=1;
IN2=0;
IN3=0;
IN4=1;
PWM_Set(255-Speed_Right,255-Speed_Left);
}
void Turn_Right(unsigned char Speed_Right,unsigned char Speed_Left)
{
IN1=0;
IN2=1;
IN3=1;
IN4=0;
PWM_Set(255-Speed_Right,255-Speed_Left);
}
void Stop(void) //剎車
{
IN1=0;
IN2=0;
IN3=0;
IN4=0;
PWM_Set(0,0);
}
void delayuus(unsigned char dela)
{
while(dela--);
}
void delayms(unsigned char nn)
{
while(nn--)
{
delayuus(245);
delayuus(245);
}
}
void test(void)
{
TH1 = 0x00;
TL1 = 0x00;
trig = 1;
delay_20us();
trig = 0;
while(!echo);
TR1 = 1;
EX1 = 1;
delayms(30);
if(flagg==1)
{ flagg=0;
dis = (TH1 * 256 + TL1);
S=(dis*1.7)/100; //算出來是CM
if(S>=700) //超出測量范圍顯示“-”
{
disbuff[0]=10; //“-”
disbuff[1]=10; //“-”
disbuff[2]=10; //“-”
}
else
{
disbuff[0]=S%1000/100;
disbuff[1]=S%1000%100/10;
disbuff[2]=S%1000%10 %10;
}
}
TH1=0;
TL1=0;
TR1=0;
EX1=0;
}
void ISR_INT1(void) interrupt 2
{
TR1=0;
EX1=0;
flagg=1;
}
void serial() interrupt 4 //中斷法
{
flag=1; //如果產生了中斷,說明單片機串口接收到數據,串口中斷標志置1
SBUF_DATA=SBUF;//將接收到的數據存放到a中
RI=0;//中斷標志
}
void time1()interrupt 1 //舵機PWM輸出
{
TH0=(65536-125)/256;
TL0=(65536-125)%256;
TimeOutCounter ++;
switch(push_val_left)
{
case 6 :
{
if( TimeOutCounter <= 7) //右:7
{
ControlPort = 1;
}
else
{
ControlPort = 0;
}
break;
}
case 7 : //
{
if( TimeOutCounter <= 16) //左
{
ControlPort = 1;
}
else
{
ControlPort = 0;
}
break;
}
case 8 : //
{
if( TimeOutCounter <= 12) //中 12:中
{
ControlPort = 1;
}
else
{
ControlPort = 0;
}
break;
}
default : break;
}
if( TimeOutCounter == 160 ) //周期20ms(理論值)
{
TimeOutCounter = 0;
}
}
void main(void)
{
unsigned char sudu=130;
PWM_ini();
//*****************************************************************************************************
TMOD=0x11;
TH1=0x00; //
TL1=0x00;
TH0=(65536-125)/256;
TL0=(65536-125)%256;
//TR0=1;
ET0=1;
//*********************************************************************************************************
SCON=0x50;// 0 1 0 1 0 0 0 0
//SM0 SM1 SM2 REN TB8 RB8 TI RI
//從上面化成二進制之后可以看到SCON的各位設置
//①SMO SM0=01 ,說明串口工作在方式1
//②REN=1,允許串口接收數據
BRT=0xfd;//9600波特率的初值
AUXR=0x11;//12T,BRTR=1啟動獨立波特率發(fā)生器
//S1BRS=1,串口1選著獨立波特率發(fā)生器作為波特率發(fā)生器
PS=1; //中斷太多,置串口為高優(yōu)先級中斷
ES=1;//允許串行中斷
EA=1;//開總中斷
IT1=1; //IT1=1表示邊沿觸發(fā)
TR0=1;
push_val_left=8;
delayms(240);
delayms(240);
TR0=0;
while(1)
{
if(SBUF_DATA==0x15)
{
TR0=0;
test(); //超聲波測距
Display(); //數碼管顯示
if(S<=10)
{
TR0=1;
push_val_left=6;
delayms(240);
delayms(240);
delayms(240);
delayms(240);
push_val_left=7;
delayms(240);
delayms(240);
delayms(240);
delayms(240);
delayms(240);
delayms(240);
TR0=0;
}
else
{
TR0=0;
test();
Display();
}
}
if(flag==1)//如果產生過中斷,證明串口接收到了數據
{ EX1=1;
flag=0;
switch(SBUF_DATA) //小車行駛方向控制 pwm
{
case leftdata: Turn_Left(sudu,sudu); break;
case rightdata: Turn_Right(sudu,sudu); break;
case forwarddata: Forward(sudu,sudu); break;
case backdata: Back(sudu,sudu); break;
case stopdata: Stop(); break;
default: break;
}
}
}
}
|