我使用ATEMGA16A做控制,PB0接485的RE/DE控制端,串口調(diào)制助手發(fā)數(shù)據(jù),一個幀的格式是24是幀頭,2C是幀尾,把中間的數(shù)據(jù)(不定長)加起來,這其中不包括2C前一個字節(jié)的數(shù)據(jù)(校驗和)。最后發(fā)送校驗和與數(shù)據(jù)長給串口。PD0+PD1是RXD和TXD。現(xiàn)在的情況是單片機根本收不到幀頭。WHY?
程序如下:
// Target : M16
// Crystal: 1.0000Mhz
#include <iom16v.h>
#include <macros.h>
#define uchar unsigned char
#define uint unsigned int
#define BAUD 4800
#define F_CPU 1000000
#define RW PORTB0
void delay(uint ms)
{
uint l,j;
for(j=0;j<ms;j++)
for(l=0;l<7373;l++); //延時1ms
}
void port_init(void) //端口初始化
{
PORTA = 0x00;
DDRA = 0x00;
PORTB = 0x00;
DDRB = 0x01;
PORTC = 0x00; //m103 output only
DDRC = 0xff;
PORTD = 0xfd;
DDRD = 0xfe;
}
void uart_init(void) //串口初始化
{
UCSRB = 0x00; //接收發(fā)送中斷禁止
// RXCIE 7 接收結(jié)束中斷使能
// TXCIE 6 發(fā)送結(jié)束中斷使能
// UDRIE 5 數(shù)據(jù)寄存器空中斷使能
// RXEN 4 接收使能
// TXEN 3 發(fā)送使能
// UCSZ2 2 與UCSZ1 UCSZ0 結(jié)合確定字符長度,011為8位
// RXB8 1 接收數(shù)據(jù)位8
// TXB8 0 發(fā)送數(shù)據(jù)位8
UCSRA = 0x00;//不倍速
// RXC 7 接收結(jié)束標(biāo)志位
// TXC 6 發(fā)送結(jié)束標(biāo)志位
// UDRE 5 數(shù)據(jù)寄存器空標(biāo)志位
// FE 4 幀錯誤標(biāo)志位,在寫UCSRA時必須寫0
// DOR 3 數(shù)據(jù)溢出標(biāo)志位,在寫UCSRA時必須寫0
// PE 2 奇偶校驗錯誤標(biāo)志位
// U2X 1 倍數(shù)發(fā)送
// MPCM 0 多處理器模式
UCSRC|= (1<<URSEL) | (1 << UCSZ1) | (1 << UCSZ0); //異步,數(shù)據(jù)格式8,沒有校驗,1位停止位
// URSEL 7 寫UCSRC時,URSEL應(yīng)設(shè)置為 1。UBRRH時為0
// UMSEL 6 0異步1同步
// UPM1 5 奇偶校驗?zāi)J剑琔PM1:UPM0,00禁止;10偶校驗;11奇校驗
// UPM0 4
// USBS 3 停止位選擇,0一位,1二位。
// UCSZ1 2
// UCSZ0 1
// UCPOL 0 用于同步模式,異步清零
UBRRL = (F_CPU / BAUD / 16 - 1) % 256; //波特率設(shè)置
UBRRH = (F_CPU / BAUD / 16 - 1) / 256;
UCSRB|= (1 << RXCIE) | (1 << RXEN) | (1 << TXEN); //接收中斷使能,接收使能,發(fā)送使能
}
uint uart_receive (void) // 等待獲得一個字節(jié)數(shù)據(jù)
{
PORTB&=~(1<<RW);
while ( !(UCSRA & (1<<RXC)) );
//UCSRA
//Bit 7 – RXC: USART 接收結(jié)束,接收緩沖器中有未讀出的數(shù)據(jù)時 RXC 置位,否則清零。
//Bit 6 – TXC: USART 發(fā)送結(jié)束,發(fā)送移位緩沖器中的數(shù)據(jù)被送出,且當(dāng)發(fā)送緩沖器 (UDR) 為空時TXC 置位。
//Bit 5 – UDRE: USART數(shù)據(jù)寄存器空UDRE標(biāo)志指出發(fā)送緩沖器(UDR)是否準(zhǔn)備好接收新數(shù)據(jù)。UDRE為1說明緩沖器為空,已準(zhǔn)備好進行數(shù)據(jù)接收。
//Bit 4 – FE: 幀錯誤
//Bit 3 – DOR: 數(shù)據(jù)溢出
//Bit 2 – PE: 奇偶校驗錯誤
//Bit 1 – U2X: 倍速發(fā)送
//Bit 0 – MPCM: 多處理器通信模式
return UDR;
}
void init_devices(void) //端口初始化
{
//stop errant interrupts until set up
CLI(); //disable all interrupts
port_init();
uart_init();
MCUCR = 0x00;
GICR = 0x00;
TIMSK = 0x00; //timer interrupt sources
SEI(); //re-enable interrupts
}
//*************串口發(fā)送字節(jié)數(shù)據(jù),調(diào)用前先使485的發(fā)送腳置高PB0,使用232串口的話,不必處理*********//
void uart_send(uint i)
{
PORTB|=(1<<RW); //485發(fā)送使能
while(!(UCSRA&(1<<UDRE))); //等待發(fā)送結(jié)束
UDR=i;
UCSRA|=0x40; //清除發(fā)送結(jié)束標(biāo)志位
delay(50);
PORTB&=~(1<<RW); //485接收使能
}
//*****************************************主程序*********************************************//
void main(void)
{
uint sum=0,overflag=0;
uint data_r1=0,data_r2=0,data0=0; //參數(shù)初始化
init_devices(); //接收、發(fā)送使能
while(data0!=0x24) //檢查是否為幀頭
{
data0=uart_receive();
}
while(data_r2!=0x2c) //檢查是否為幀尾
{
data_r2=uart_receive();
sum=sum+data_r1;
data_r1=data_r2;
overflag=overflag+1;
}
overflag=overflag-1;
sum=sum%0xff;
uart_send(sum);
uart_send(overflag);
}
發(fā)送數(shù)據(jù)格式為 24 01 02 03 2C,沒回數(shù)
幫頂上!!
| 歡迎光臨 (http://m.raoushi.com/bbs/) | Powered by Discuz! X3.1 |