智能寻轨器 行走调试方法

智能寻轨器 行走调试方法,第1张

1 引言

智能汽车是汽车电子、人工智能、模式识别、自动控制、计算机、机械多个学科领域的交叉综合的体现,具有重要的应用价值。智能寻迹车是基于飞思卡尔MC9S12DGl28单片机开发实现的,该系统采用CCD传感器识别道路中央黑色的引导线,利用传感器检测智能车的加速度和速度,在此基础上利用合理的算法控制智能车运动,从而实现快速稳定的寻迹行驶。

2 硬件系统设计

该系统硬件设计主要由MC9S12DGl28控制核心、电源管理模块、直流电机驱动模块、转向舵机控制模块、道路信息检测模块、速度检测模块和加速度检测模块等组成,其结构框图如图1所示。

图片看不清楚?请点击这里查看原图(大图)。

2.1 主控制器模块

智能车的控制核心为MC9S12DGl28。MC9S12DGl28是飞思卡尔公司生产的一款16位单片机,片内总线时钟可达到25 MHz;片内资源包括8 K RAM、128 K Flash、2 K EEP-ROM;SCI,SPI,PWM和串行接口模块;脉宽调制模块(PWM)可设置成4路8位或2路16位,逻辑时钟选择频率脉宽:2个8路10位A/D转换器,增强型捕捉定时器并支持背景调试模式等。

2.2 电源管理模块

该系统设计采码简神用7.2 V/1 800 mA镍镉电池供电,7.2 V电压经过BMlll7—5稳压后得到5 V电压,向单片机、红外传感器和加速度传感器供电。5 V电压经MAX8715升压后得到12 V电压,向CCD图像传感器供电:7.2 V电压经二极管降压得到约咐困6 V的电压来驱动舵机;电机驱动器MC33886则直接由电源供电。

2.3 道路信息检测模块

该系统设计使用NEC公司的线阵CCDμPD3575D检测赛道信息。该器件可工作在5 V驱动(脉冲)和12 V电源条件下。μPD3575D的驱动需要4路脉冲,分别为转移栅时钟φIO、复位时钟φRD、采样保持时钟φSHO和传输门时钟φTG。系统设计由外围电路直接产生CCD驱动时钟,采用计数器和触发器专门设计时序电路,产生转移栅时钟φIO、复位时钟φRO、采样保持时钟φSHO,单片机只需产生一个帧同步信号(传输门信号φTG)与外围时序电路保持同步即可。μPD3575D输出的是模拟信号,将采集图像传输至单片机,一般需对μPD3575D输出信号进行A/D转换,考虑到设计实际上只需要区分黑色和白色,μPD3575D对这两种输出信号差异较大迟亏,因此,将μPD3575D输出信号放大后直接使用一个比较器对信号二值化处理,如图2所示。由单片机检测二值信号的跳变时间,便可计算出黑线位置,从而进一步缩短单片机在CCD上所消耗的时间。

我做小车用的就是这个程序,好使,选为最佳答案就是。多谢

#include "reg52.h"

#define det_Dist 2.55 //单个脉冲对应的小车行走距离,其值为车轮周长/4

#define RD 9 //小车对角轴长度

#define PI 3.1415926

#define ANG_90 90

#define ANG_90_T 102

#define ANG_180 189

/*============================全局变量定义区============================*/

sbit P10=P1^0//控制继电器的开闭

sbit P11=P1^1//控制金属接近开关

sbit P12=P1^2//控制颜色传感器的开闭

sbit P07=P0^7//控制声光信号的开启

sbit P26=P2^6//接收颜色传感器的信号,白为0,黑为1

sbit P24=P2^4//左

sbit P25=P2^5//右 接收左右光传感器的信号,有光为0

unsigned char mType=0//尘埋设置运动的方式,0 向前 1 向左 2 向后 3 向右

unsigned char Direction=0//小车的即时朝向 0 朝上 1 朝左 2 朝下 3 朝右

unsigned sX=50unsigned char sY=0//小车的相对右下角的坐标 CM(sX,sY)

unsigned char StartTask=0//获得铁片后开始执行返回卸货任务,StartTask置一

unsigned char Inter_EX0=0// 完成一个完整的任务期间只能有一次外部中断

// Inter_EX0记录外部中断0的中断状态

// 0 动作最近的前一次未中断过,

// 1 动作最近的前一次中断过

unsigned char cntIorn=0//铁片数

unsigned char bkAim=2//回程目的地,0为A仓库,1为B仓库,2为停车场,

//(在MAIN中接受铁片颜凳漏色判断传感器的信号来赋值)

unsigned char Light_Flag=0//进入光引导区枣兄烂的标志(1)

unsigned int cntTime_5Min=0//时间周期数,用于 T0 精确定时

unsigned int cntTime_Plues=0//霍尔开关产生的脉冲数

/*============================全局变量定义区============================*/

/*------------------------------------------------*/

/*-----------------通用延迟程序-------------------*/

/*------------------------------------------------*/

void delay(unsigned int time) // time*0.5ms延时

{

unsigned int i,j

for(j=0j<timej++)

{

for(i=0i<60i++)

{}

}

}

/*-----------------------------------------------*/

/*-------------------显示控制模块----------------*/

/*-----------------------------------------------*/

/*数码管显示,显示铁片的数目(设接在P0,共阴)*/

void Display(unsigned char n)

{

char Numb[12]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x37,0x77}

P0=Numb[n]

}

/*-----------------------------------------------*/

/*-------------------传感器模块------------------*/

/*-----------------------------------------------*/

/*光源检测程序: */

/*用于纠正小车运行路线的正确性*/

unsigned char LightSeek()

{ void Display(unsigned char)

bit l,r

l=P24

r=P25

if(l==0&&r==1)

{

//Display(1)

return (3)//偏左,向右开

}

if(r==0&&l==1)

{

//Display(3)

return(1)//偏右,向左开

}

if((l==1&&r==1)||(l==0&&r==0))

{//Display(9)

return(0)//没有偏离,前进

}

}

/*铁片检测程序: */

/*判断铁片的颜色,设定bkAim,0为A仓库,1为B仓库,2为停车场*/

void IornColor()

{

delay(4000)

bkAim=(int)(P26)

Display((int)(P26)+2)

}

/*-----------------------------------------------*/

/*------------------运动控制模块-----------------*/

/*-----------------------------------------------*/

/*====基本动作层:完成基本运动动作的程序集====*/

/*运动调整程序: */

/*对小车的运动进行微调*/

void ctrMotor_Adjust(unsigned char t)

{

if(t==0)

{

P2=P2&240|11//用来解决两电机不对称的问题

delay(6)

}

if(t==3)

{

P2=P2&250//向左走

delay(1)

}

if(t==1)

{

P2=(P2&245)

delay(1)//向右走

}

P2=((P2&240)|15)

delay(10)

}

/*直走程序:*/

/*控制小车运动距离,dist为运动距离(cm),type为运动方式(0 2)*/

/*只改变小车sX 和 sY的值而不改变Direction的值. */

void ctrMotor_Dist(float dist,unsigned char type)

{unsigned char t=0

mType=type

P2=((P2&240)|15)

cntTime_Plues=(int)(dist/det_Dist)

while(cntTime_Plues)

{

if(Inter_EX0==1&&StartTask==0)

{

cntTime_Plues=0

break

}

if(Light_Flag==1) t=LightSeek()

if(type==0) //向前走

{

P2=P2&249

delay(40)

ctrMotor_Adjust(t)

}

if(type==2) //向后退

{

P2=P2&246

delay(50)

ctrMotor_Adjust(t)

}

P2=((P2&240)|15)

if(mType==2) delay(60)//刹车制动 0.5ms

else delay(75)

}

}

/*拐弯程序: */

/*控制小车运动角度,type为运动方式(1 3) */

/*只改变小车Direction的值而不改变sX 和 sY的值*/

void ctrMotor_Ang(unsigned char ang,unsigned char type,unsigned char dir)

{

unsigned char i=0

mType=type

P2=((P2&240)|15)

cntTime_Plues=(int)((PI*RD*90/(180*det_Dist)*1.2)*ang/90)

while(cntTime_Plues)

{

if(Inter_EX0==1&&StartTask==0)

{

cntTime_Plues=0

break

}

if(type==1) //向左走

{

P2=P2&250

delay(100)

ctrMotor_Adjust(0)

}

if(type==3) //向右走

{

P2=P2&245

delay(100)

ctrMotor_Adjust(0)

}

P2=((P2&240)|15)

delay(50)//刹车制动 0.5ms

}

if(!(Inter_EX0==1&&StartTask==0))

{

Direction=dir

}

}

/*====基本路线层:描述小车基本运动路线的程序集====*/

/*当小车到达仓库或停车场时,放下铁片或停车(0,1为仓库,2为停车场)*/

void rchPlace()

{unsigned int time,b,s,g

time=(int)(cntTime_5Min*0.065535)//只有一个数码管时,轮流显示全过程秒数 个 十 百

b=time%100

s=(time-b*100)%100

g=(time-b*100-s*10)%10

if(bkAim==2)

{

//到达停车场了,停车

EA=0

P2=((P2&240)|15)

while(1)

{

Display(10)//N

delay(2000)

Display(cntIorn)

delay(2000)

Display(11)//A

delay(2000)

Display(b)

delay(2000)

Display(s)

delay(2000)

Display(g)

delay(2000)

}

}

else

{

if(Inter_EX0==1&&StartTask==1)P10=0//到达仓库,卸下铁片

}

}

/*无任务模式: */

/*设置小车的固定运动路线,未发现铁片时的运动路线*/

void BasicRoute()

{ //Light_Flag=1

ctrMotor_Dist(153,0)

//Light_Flag=0

ctrMotor_Ang(ANG_90,1,1)

ctrMotor_Dist(100-sX,0)

ctrMotor_Dist(125,2)

ctrMotor_Dist(73,0)

ctrMotor_Ang(ANG_90,1,2)

//Light_Flag=1

ctrMotor_Dist(153,0)

//Light_Flag=0

ctrMotor_Ang(ANG_180,1,0)

rchPlace()

}

/*任务模式: */

/*设置小车的发现铁片后的运动路线*/

void TaskRoute()

{

//基本运行路线表,记载拐弯 0 向前 1 左拐 2 向后 3 右拐,正读去A区反读去B区

StartTask=1

ctrMotor_Ang(ANG_90_T,1,2)

if(bkAim==1)//仓库A

{

ctrMotor_Dist(10,0)

P2=((P2&240)|15)

delay(60)

ctrMotor_Ang(ANG_90_T,1,3)

ctrMotor_Dist(100-sX,2)

ctrMotor_Ang(ANG_90_T,1,2)

Light_Flag=1

ctrMotor_Dist(153,2)

Light_Flag=0

// ctrMotor_Ang(208,1,0)

}

else if(bkAim==0) //仓库B

{

ctrMotor_Dist(10,0)

P2=((P2&240)|15)

delay(60)

ctrMotor_Ang(ANG_90_T,1,3)

ctrMotor_Dist(100-sX,0)

ctrMotor_Ang(ANG_90_T,1,0)

Light_Flag=1

ctrMotor_Dist(153,2)

Light_Flag=0

//ctrMotor_Ang(208,1,0)

}

delay(5000)

rchPlace()

}

/*---------------------------------------------*/

/*-------------------主程序段------------------*/

/*---------------------------------------------*/

void main()

{

delay(4000)

P2=0xff//初始化端口

P07=0

P1=0

TMOD=0x01//初始化定时器0/1 及其中断

TL0=0

TH0=0

TR0=1

ET0=1

ET1=1

IT0=1//初始化外部中断

EX0=1

IT1=1

EX1=1

EA=1

P11=1

while(1)

{

Display(cntIorn)

bkAim=2

BasicRoute()

if(Inter_EX0==1)

{

TaskRoute()//按获得铁片后的路线运动

IE0=0

EX0=1

}

Inter_EX0=0

}

}

/*----------------------------------------------------*/

/*----------------------中断程序段--------------------*/

/*----------------------------------------------------*/

/*定时器0中断程序: */

/*当时间过了5分钟,则就地停车并进入休眠状态*/

void tmOver(void) interrupt 1

{

cntTime_5Min++

TL0=0

TH0=0

if(cntTime_5Min>=4520)

{

Display(5)

P2=((P2&240)|15)

EA=0//停车程序

P07=1

delay(4000)

PCON=0X00

while(1)

}

}

/*外部中断0中断程序: */

/*发现铁片,发出声光信号并将铁片吸起,发光二极管和蜂鸣器*/

/*并联在一起(设接在P07). 0为A仓库,1为B仓库,2为停车场*/

void fndIorn(void) interrupt 0

{

unsigned char i

P10=1

P2=((P2&240)|15)//停车

P07=1

delay(1000)//刹车制动 0.5ms

P07=0

Inter_EX0=1

cntIorn++

Display(cntIorn)

for(i=0i<40i++)

{

P2=P2&249

delay(2)

P2=((P2&240)|15)

delay(2)

}

P2=P2&249

delay(100)

P2=((P2&240)|15)//停车

IornColor()//判断铁片黑白,设置bkAim

for(i=0i<95i++)

{

P2=P2&249

delay(3)

P2=((P2&240)|15)

delay(2)

}

P2=((P2&240)|15)//停车

delay(4000)//把铁片吸起来

EX0=0

}

/*外部中断1中断程序:*/

/*对霍尔开关的脉冲记数,对小车的位置进行记录,以便对小车进行定位*/

void stpMove(void) interrupt 2

{

cntTime_Plues--

if(Direction==0) //向上

{

if(mType==0) sY+=det_Dist

else if(mType==2)

sY-=det_Dist

}

else if(Direction==1) //向左

{

if(mType==0) sX+=det_Dist

else if(mType==2)

sX-=det_Dist

}

else if(Direction==2) //向下

{

if(mType==0) sY-=det_Dist

else if(mType==2)

sY+=det_Dist

}

else if(Direction==3) //向右

{

if(mType==0) sX-=det_Dist

else if(mType==2)

sX+=det_Dist

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12399855.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-25
下一篇 2023-05-25

发表评论

登录后才能评论

评论列表(0条)

保存