【程序已測試】51單片機(jī)控制sim900a
掃描二維碼
隨時(shí)隨地手機(jī)看文章
已經(jīng)經(jīng)過測試,直接上程序,不懂@me
#include"reg52.h"
#include"string.h"
#define uchar unsigned char
#define uint unsigned int
//以下是板子上LED的配置,把Px_x改成自己對應(yīng)的腳。
#define LED1_ON P3_2 = 1
#define LED1_OFF P3_2 = 0
#define LED2_ON P3_3 = 1
#define LED2_OFF P3_3 = 0
#define LED3_ON P3_4 = 1
#define LED3_OFF P3_4 = 0
#define LED4_ON P3_5 = 1
#define LED4_OFF P3_5 = 0
//以下是你的51單片機(jī)的晶振大小,使用
#define FOSC_110592M
//#define FOSC_12M
//以下是開機(jī)后撥打的手機(jī)號(hào),改成自己想要打的號(hào)碼即可。
uchar NUM[] = "15296811438";
uchar GsmRcv[50] = {0};
uchar GsmRcvAt[50] = {0};
uchar GsmRcvCnt = 0;
uchar GsmAtFlag = 0;
//以下修改對應(yīng)的手機(jī)卡服務(wù)商,中國移動(dòng)或者中國聯(lián)通。如果是中國聯(lián)通,就把第一個(gè)注釋掉。打開第二個(gè)。
uchar SER[] = "CHINA MOBILE";
//uchar SER[] = "CHINA UNICOM";
void SerialInti()//初始化程序(必須使用,否則無法收發(fā))
{
TMOD=0x20;//定時(shí)器1操作模式2:8位自動(dòng)重載定時(shí)器
#ifdef FOSC_12M //在這里根據(jù)晶振大小設(shè)置不同的數(shù)值初始化串口
TH1=0xf3;//裝入初值,波特率2400
TL1=0xf3;
#else
TH1=0xfd;//裝入初值,波特率9600
TL1=0xfd;
#endif //end of SOC_12M
TR1=1;//打開定時(shí)器
SM0=0;//設(shè)置串行通訊工作模式,(10為一部發(fā)送,波特率可變,由定時(shí)器1的溢出率控制)
SM1=1;//(同上)在此模式下,定時(shí)器溢出一次就發(fā)送一個(gè)位的數(shù)據(jù)
REN=1;//串行接收允許位(要先設(shè)置sm0sm1再開串行允許)
EA=1;//開總中斷
ES=1;//開串行口中斷
}
//注意,無論接收到信號(hào)還是發(fā)送完信號(hào),都會(huì)進(jìn)中斷服務(wù)程序的
void Serial_interrupt() interrupt 4
{
uchar i = 0;
if(RI == 1) //收到信息
{
RI=0;//接收中斷信號(hào)清零,表示將繼續(xù)接收
GsmRcv[GsmRcvCnt] = SBUF;
// Uart1Send(tmp);
GsmRcvCnt++;
//收到了完整的AT指令,完整的AT指令是以0x0a 0x0d結(jié)尾的。故作此判斷,在接收的過程中是否收到0x0a 0x0d
if(GsmRcv[GsmRcvCnt-2] == 0x0d && GsmRcv[GsmRcvCnt-1] == 0x0a && GsmRcvCnt >= 2)
{
//一旦收到0x0a 0x0d,就將數(shù)據(jù)保存起來。用戶主函數(shù)的判斷。
for(i=0; i<>
{
GsmRcvAt[i] = GsmRcv[i];
GsmRcv[i] = 0;
}
GsmRcvCnt = 0;
GsmAtFlag = 1;//收到了完整的at指令,通過這個(gè)標(biāo)志位置1,這樣主函數(shù)就知道去判斷了。
}
else if(GsmRcvCnt >= 50)//因?yàn)閮?nèi)存有限,收到了50個(gè)字符還是沒有看到0x0a 0x0d的話,就重新開始接收吧。
{
GsmRcvCnt = 0;
}
}
}
//串行口連續(xù)發(fā)送char型數(shù)組,遇到終止號(hào)/0將停止
void Uart1Sends(uchar *str)
{
while(*str!='�')
{
SBUF=*str;
while(!TI);//等待發(fā)送完成信號(hào)(TI=1)出現(xiàn)
TI=0;
str++;
}
}
//延時(shí)函數(shù)大概是1s鐘,不過延時(shí)大的話不準(zhǔn)...
void DelaySec(int sec)
{
uint i , j= 0;
for(i=0; i<>
{
for(j=0; j<65535; j++)
{
}
}
}
void main()
{
uint i,j = 0;
LED1_OFF;
LED2_OFF;
LED3_OFF;
LED4_OFF;
SerialInti();
P2_4 = 0;
//判斷是否啟動(dòng)完成
GsmAtFlag = 0;
while(GsmAtFlag == 0)
{
Uart1Sends("atir");//設(shè)置sim300波特率
LED1_ON;
DelaySec(1);//延時(shí)1秒
LED1_OFF;
}
GsmAtFlag = 0;
LED1_OFF;
//檢測信號(hào)
while(1)
{
Uart1Sends("AT+COPS?r");
LED2_ON;
DelaySec(1);//延時(shí)1秒
LED2_OFF;
while(GsmAtFlag == 0);
if(strstr(GsmRcvAt, SER) )//檢測是否收到 CHINA MOBILE 服務(wù)商信息。如果收到證明是連接上網(wǎng)絡(luò)了
{
break;
}
}
LED2_OFF;
//來電提示
while(1)
{
//如果有來電,模塊返回RING,此時(shí)發(fā)送AT+CLCC/r去讀取來電號(hào)碼
if(strstr(GsmRcvAt, "RING"))
{
Uart1Sends("AT+CLCCr");//發(fā)送AT+CLCC/r去讀取來電號(hào)碼
}
if(strstr(GsmRcvAt, "+CLCC") && strstr(GsmRcvAt, NUM) )//檢測是否收到來電顯示和對應(yīng)的來電號(hào)碼。
{
LED2_ON;
}
if(strstr(GsmRcvAt, "NO CARRIER"))//檢測對方是否掛斷電話。這個(gè)指令會(huì)等待對方掛斷了電話數(shù)秒后執(zhí)行。
{
LED2_OFF;
}
//一定的時(shí)間讓LED閃爍,以觀察系統(tǒng)運(yùn)行是否正常。
i++;
if(i <= 500)
{
LED1_ON;
}
else if(i > 500 && i < 1000)
{
LED1_OFF;
}
else
{
i = 0;
}
}
}