用到了兩塊51單片機,一塊遙控,另一塊控制小車。小車上測得的距離傳送到遙控的單片機上顯示。屏幕用的是1602;
單片機源程序如下:
- #include <reg52.h>
- #include <intrins.h>
- #include <api.h>
- #include <duoji.h>
- #define uchar unsigned char
- #define uint unsigned int
- #define TX_ADR_WIDTH 5 // 5字節寬度的發送/接收地址
- #define TX_PLOAD_WIDTH 4 // 數據通道有效數據寬度
- unsigned long ultrasonic[5] = {0};
- unsigned int time=0;//用于存放定時器時間值
- unsigned long S = 0;//用于存放距離的值//用于超聲波測距
- uchar flag;
- static uint aaa;//接受數據
- static uint dd;//存放距離
- uchar code TX_ADDRESS[TX_ADR_WIDTH] = {0x34,0x43,0x10,0x10,0x01}; // 定義一個靜態發送地址
- uchar RX_BUF[TX_PLOAD_WIDTH];
- uchar TX_BUF[TX_PLOAD_WIDTH];
- uchar DATA = 0x01;
- uchar bdata sta;
- sbit RX_DR = sta^6;
- sbit TX_DS = sta^5;
- sbit MAX_RT = sta^4;
- sbit TX=P3^7; //超聲波模塊Trig 控制端
- sbit RX=P3^6; //超聲波模塊Echo 接收端
- sbit hw1=P2^6;
- sbit hw2=P2^7;//紅外循跡
- bit fl =0; //量程溢出標志位
- void yaokong2(void);
- void init_io(void)
- {
- CE = 0; // 待機
- CSN = 1; // SPI禁止
- SCK = 0; // SPI時鐘置低
- IRQ = 1; // 中斷復位
- }
- void delay_ms(uchar x)
- {
- uchar i, j;
- i = 0;
- for(i=0; i<x; i++)
- {
- j = 250;
- while(--j);
- j = 250;
- while(--j);
- }
- }
- uchar SPI_RW(uchar byte)
- {
- uchar i;
- for(i=0; i<8; i++) // 循環8次
- {
- MOSI = (byte & 0x80); // byte最高位輸出到MOSI
- byte <<= 1; // 低一位移位到最高位
- SCK = 1; // 拉高SCK,nRF24L01從MOSI讀入1位數據,同時從MISO輸出1位數據
- byte |= MISO; // 讀MISO到byte最低位
- SCK = 0; // SCK置低
- }
- return(byte); // 返回讀出的一字節
- }
- uchar SPI_RW_Reg(uchar reg, uchar value)
- {
- uchar status;
- CSN = 0; // CSN置低,開始傳輸數據
- status = SPI_RW(reg); // 選擇寄存器,同時返回狀態字
- SPI_RW(value); // 然后寫數據到該寄存器
- CSN = 1; // CSN拉高,結束數據傳輸
- return(status); // 返回狀態寄存器
- }
- uchar SPI_Read(uchar reg)
- {
- uchar reg_val;
- CSN = 0; // CSN置低,開始傳輸數據
- SPI_RW(reg); // 選擇寄存器
- reg_val = SPI_RW(0); // 然后從該寄存器讀數據
- CSN = 1; // CSN拉高,結束數據傳輸
- return(reg_val); // 返回寄存器數據
- }
- uchar SPI_Read_Buf(uchar reg, uchar * pBuf, uchar bytes)
- {
- uchar status, i;
- CSN = 0; // CSN置低,開始傳輸數據
- status = SPI_RW(reg); // 選擇寄存器,同時返回狀態字
- for(i=0; i<bytes; i++)
- pBuf[i] = SPI_RW(0); // 逐個字節從nRF24L01讀出
- CSN = 1; // CSN拉高,結束數據傳輸
- return(status); // 返回狀態寄存器
- }
- uchar SPI_Write_Buf(uchar reg, uchar * pBuf, uchar bytes)
- {
- uchar status, i;
- CSN = 0; // CSN置低,開始傳輸數據
- status = SPI_RW(reg); // 選擇寄存器,同時返回狀態字
- for(i=0; i<bytes; i++)
- SPI_RW(pBuf[i]); // 逐個字節寫入nRF24L01
- CSN = 1; // CSN拉高,結束數據傳輸
- return(status); // 返回狀態寄存器
- }
- void RX_Mode(void)
- {
- CE = 0;
- SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 接收設備接收通道0使用和發送設備相同的發送地址
- SPI_RW_Reg(WRITE_REG + EN_AA, 0x01); // 使能接收通道0自動應答
- SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01); // 使能接收通道0
- SPI_RW_Reg(WRITE_REG + RF_CH, 40); // 選擇射頻通道0x40
- SPI_RW_Reg(WRITE_REG + RX_PW_P0, TX_PLOAD_WIDTH); // 接收通道0選擇和發送通道相同有效數據寬度
- SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07); // 數據傳輸率1Mbps,發射功率0dBm,低噪聲放大器增益
- SPI_RW_Reg(WRITE_REG + CONFIG, 0x0f); // CRC使能,16位CRC校驗,上電,接收模式
- CE = 1; // 拉高CE啟動接收設備
- }
- void TX_Mode(uchar * BUF)
- {
- CE = 0;
- SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH); // 寫入發送地址
- SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 為了應答接收設備,接收通道0地址和發送地址相同
- SPI_Write_Buf(WR_TX_PLOAD, BUF, TX_PLOAD_WIDTH); // 寫數據包到TX FIFO
- SPI_RW_Reg(WRITE_REG + EN_AA, 0x01); // 使能接收通道0自動應答
- SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01); // 使能接收通道0
- SPI_RW_Reg(WRITE_REG + SETUP_RETR, 0x0a); // 自動重發延時等待250us+86us,自動重發10次
- SPI_RW_Reg(WRITE_REG + RF_CH, 40); // 選擇射頻通道0x40
- SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07); // 數據傳輸率1Mbps,發射功率0dBm,低噪聲放大器增益
- SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e); // CRC使能,16位CRC校驗,上電
- CE = 1;
- }
- uchar Check_ACK(bit clear)
- {
- while(IRQ);
- sta = SPI_RW(NOP); // 返回狀態寄存器
- if(MAX_RT)
- if(clear) // 是否清除TX FIFO,沒有清除在復位MAX_RT中斷標志后重發
- SPI_RW(FLUSH_TX);
- SPI_RW_Reg(WRITE_REG + STATUS, sta); // 清除TX_DS或MAX_RT中斷標志
- IRQ = 1;
- if(TX_DS)
- return(0x00);
- else
- return(0xff);
- }
- void delay__(uint count) //delay
- {
- uint i;
- while(count)
- {
- i=200;
- while(i>0)
- i--;
- count--;
- }
- }
- void Delay10us(unsigned int i) //10us延時函數
- {
- unsigned int j;
- do{
- j = 10;
- do
- {
- _nop_();
- }while(--j);
- }while(--i);
- }
- void StartModule() //啟動超聲波模塊
- {
- TX=1; //啟動一次模塊
- Delay10us(2);
- TX=0;
- }
- unsigned long conut1(void) //測量一次距離
- {
- unsigned long dis = 0;
- time=TH1*256+TL1;
- TH1=0;
- TL1=0;
-
- dis=(float)(time*1.085)*0.17;
- return dis;
- }
- void bubble(unsigned long *a, int n) //冒泡程序
- {
- int i,j,temp;
- for(i = 0;i < n-1; i++)
- {
- for(j = i + 1; j < n; j++)
- {
- if(a[i] > a[j])
- {
- temp = a[i];
- a[i] = a[j];
- a[j] = temp;
- }
- }
- }
- }
- uint distance22()
- {
- uint dis;
- RX = 1;
- StartModule();
- while(RX==0);
-
- TR1=1;
- while(RX);
- TR1=0;
- dis=conut1();
- return dis;
- }
- unsigned long Distance(void) //測距
- {
- int num = 0;
- unsigned long Distance = 0;
- RX = 1;
- StartModule();
- while(RX==0);
-
- TR1=1;
- while(RX);
- TR1=0;
-
- conut1();
- while(num < 5) // 測5次數據
- {
- RX = 1;
- StartModule();
- while(RX==0);
-
- TR1=1;
- while(RX);
- TR1=0;
-
- S = conut1();
- ultrasonic[num] = S;
- num++;
- }
- num = 0;
- bubble(ultrasonic, 5);
- Distance = (ultrasonic[1]+ultrasonic[2]+ultrasonic[3])/3;
- return Distance;
- }
- void duobi(void) //避障程序
- {
- uint a[]={0,0};//zuoyoujuli
- // dd=Distance();
- // juli=distance;
- z0();
- if(dd>600)//old接收到的為cm,distance為毫米
- {
- qj();
- }//前進時發射距離
- else
- {
- t(10);
- y90(); y90(); y90();
- delaydj(100);
- a[1]=Distance();
- z90(); z90(); z90();
- delaydj(100);
- a[0]=Distance();
- z0();
- if (a[0]>a[1] && a[1]>300)
- {
- z(900);
- t(50);
- }
- else if(a[0]<a[1] && a[0]>300)
- {
- y(900);
- t(50);
- }
- else if (a[0]>a[1] && a[1]<300 || a[0]<a[1] && a[1]<300)
- {
-
- ht(500);
- t(50);
- if(a[0]<a[1])
- y(820);
- else z(820);
- }
- else
- {
- if(a[0]<a[1])
- {hz(800);
- t(10);}
- else hy(800);
-
- }
- }
- }
- //void hongwai(void)
- //{
- // EX0 = 1;//開外部中斷0
- // IT0 = 1;//外部中斷0高電平觸發
- // EX1 = 1;//開外部中斷0
- // IT1 = 1;//外部中斷0高電平觸發
- // switch (hms)
- // {
- // case 0:qjj(100); break;
- // case 1:{t(10);z(200);}break;
- // case 2:{t(10);y(200);}break;
- // }
- // if(dd<200)
- // duobi();
- // hms=0;
- //}
- void fashe (uint b) //發射數據十六進制碼
- {
- b/=10; //轉化為cm
- TX_BUF[2]=TT;
- if(fl!=1 && b<500 && b>250)
- {
- b-=250;
- b=0x01/0x01*b;
- TX_BUF[0]=250;
- TX_BUF[1]=b;
- TX_Mode(TX_BUF);
- SPI_RW(FLUSH_TX);
- SPI_RW_Reg(WRITE_REG + STATUS, sta); // 清除TX_DS或MAX_RT中斷標志
- IRQ = 1;
- }
- else if(b<250)
- {
- b=0x01/0x01*b;
- TX_BUF[0]=b;
- TX_BUF[1]=0x00;
- TX_Mode(TX_BUF);
- SPI_RW(FLUSH_TX);
- SPI_RW_Reg(WRITE_REG + STATUS, sta); // 清除TX_DS或MAX_RT中斷標志
- IRQ = 1;
- }
- else if(fl==1)
- {
- fl=0;
- TX_BUF[0]=0x01;
- TX_BUF[1]=0x00;
- TX_Mode(TX_BUF);
- SPI_RW(FLUSH_TX);
- SPI_RW_Reg(WRITE_REG + STATUS, sta); // 清除TX_DS或MAX_RT中斷標志
- IRQ = 1;
- }
- SPI_RW_Reg(FLUSH_TX, 0x00); //清除寄存器
- init_io();
- RX_Mode();
-
- }
- uint jieshou(void)
- {
- sta = SPI_Read(STATUS); // 讀狀態寄存器
- if(RX_DR) // 判斷是否接受到數據
- {
- SPI_Read_Buf(RD_RX_PLOAD, RX_BUF, TX_PLOAD_WIDTH); // 從RX FIFO讀出數據
- flag = 1;
- }
- SPI_RW_Reg(WRITE_REG + STATUS, sta); // 清除RX_DS中斷標志
- if(flag) // 接受完成
- {
- flag = 0; // 清標志
- aaa=RX_BUF[0];
- // if(aaa>15) old=aaa; //修改避障距離
- }
- SPI_RW_Reg(FLUSH_RX, 0xff);//清除寄存器
- return aaa;
- }
- void yaokong(uint a)
- {
- switch (a)
- {
- case 0: hz(300); break;
- case 1: qjj(500); break;
- case 2: hy(300); break;
- case 3: duobi(); break;
- case 4: z(400); break;
- case 5: ht(400);break;
- case 6: y(500); break;
- case 8: yaokong2();break;
- case 15: t(1); break;
- // case 7: hongwai();break;
- }
- }
- void yaokong2(void)
- {
- while(aaa!=15)//接收到3退出
- { TT=9;
- RX_Mode();
- aaa=jieshou();
- switch(aaa)
- {
- case 1:qjj(500);break;
- case 4:z(300); break;
- case 5:ht(500);break;
- case 6:y(500);break;
- case 0:t(1);break;
- // case 9:{if(TT<9)++TT;}break;
- // case 10:{if(TT>4)--TT;}break;//加減速
- }
- if(aaa!=15)
- aaa=0;
- }
- }
- //void gensui(void)
- //{
- // uint pa;
- // TT=4;
- // if(hw1==0 && hw2==0)
- // {
- // dd=Distance();
- // if(dd>30)
- // { pa=(dd/10-30)*25;qjj(pa);}
- // else {pa=(30-dd/10)*25;ht(pa);}
- //
- // }
- // else if(hw1==1 && hw2==1)
- // {
- // dd=Distance();
- // if(dd>250 && dd<350)
- // t(1);
- // else
- // {
- // if (dd>300)
- // { pa=(dd/10-30)*25;qjj(pa);}
- // else
- // {pa=(30-dd/10)*25;ht(pa);}
- // }
- // }
- // else if(hw1==0 && hw2==1)
- // y(400);
- // else if(hw1==1 && hw2==0)
- // z(400);
- //}
- void sudu(void)
- {
- if(dd>1000) {TT=9;}
- else if(dd>400&&dd<1000) {TT=dd/100;}
- else TT=4;
- }
- void timer1() interrupt 3 //T1 中斷用來計數器溢出,超過測距范圍
- {
- fl=1; //中斷溢出標志
- RX=0;
- }
- void timer0(void) interrupt 1//定時器0中斷函數
- {
- TR0 = 0; //關閉定時器0
- TH0 = 0xff; //重裝初值0.1ms
- TL0 = 0xa3;
- kk++;
- if(kk>=10)
- {
- kk=0;
- ……………………
- …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼
捕獲.JPG (73.14 KB, 下載次數: 79)
下載附件
2020-3-23 21:36 上傳
所有資料51hei提供下載:
遙控小車.zip
(94.59 KB, 下載次數: 52)
2020-3-23 21:36 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|