/*******************************************************************
串口1DMA接收不定長度數據:利用串口1中斷,定時器0超時判定,很好地實現串口1DMA接收
不定長度數據到XARM, 并很好地實現接收完成后,原樣返回給PC,不受256字節限制,
1-N個字節隨意(但要在Dshow定義范圍內)。
問題是:UART1_DMA死活進不了中斷 。按理說串口1DMA接收完定義的字節長度,進中斷才是正招啊。
雖然下面的方案也很好地解決了問題,但心里總不舒服,哪位大神指點下啊。單片機STC8H8K64U.
*********************************************************************/
單片機源程序如下:
//#include "dma.h"
#include "STC8h.h"
#define u32 unsigned long
#define u16 unsigned int
#define u8 unsigned char
#define MAIN_Fosc 22118400UL
#define Baudrate1 115200UL // 波特率115200
#define Timer0_Reload (65536UL - (MAIN_Fosc /1000UL))
#define DMA_AMT_LEN 255 //0-255 最大定義范圍255。實際可連續發送N多個
bit B_1ms,TANG;
u16 Rx_cnt;
u8 RX1_TimeOut;
void GPIO_Config();
void UART1_Config(u8 brt);
void DMA_Config(void);
void Timer0_Init();
void DMA_RX(u8 g);
void DMA_TX();
void UartSendByte(u8 dat);
void UartSendStr(u8 *str);
u8 xdata Dshow[4096]={0};
void main(void)
{ u8 g=0; //用于累加多少個DMA_AMT_LEN數
GPIO_Config();
DMA_Config();
UART1_Config(1);
Timer0_Init();
B_1ms=0;TANG=0;Rx_cnt=0; RX1_TimeOut=0;
while (1)
{ if(TANG){g++; DMA_RX(g) ; Rx_cnt=0; TANG=0;}//就是這樣處理類似dma中斷的.郁悶啊!???
if(B_1ms) //1ms 到
{ B_1ms = 0;
if(RX1_TimeOut > 0)
{ if(--RX1_TimeOut == 0) // 接收超時計數
{ DMA_UR1R_CR = 0x00; // 關閉 UART1_DMA
Dshow[g*(DMA_AMT_LEN+1)+Rx_cnt]='\0';
UartSendStr(Dshow);//發送到串口。當然也可用DMA發送,但涉及到字節多少的處理,偷一個懶
g =0;Rx_cnt=0; TANG=0;
DMA_RX(0); //重新初始化
}
}
}
}
}
/*********************************************************
***********************************************************/
void DMA_RX(u8 g)
{ DMA_UR1R_CFG = 0x00; // 屏蔽串口1DMA接收中斷
DMA_UR1R_STA = 0x00; //關閉dma 清零串口1DMA接收完成中斷標志、清零數據丟棄中斷標志
DMA_UR1R_AMT = DMA_AMT_LEN; // 設置傳輸總字節數(低8位):n+1
DMA_UR1R_RXA = (u16)Dshow+ (DMA_AMT_LEN+1)*g ; // 設置傳輸數據的目標地址,低8位
DMA_UR1R_CR = 0xa1;
}
/*********************************************************
***********************************************************/
void DMA_TX()
{ DMA_UR1T_CFG = 0x00; // 屏蔽串口1DMA接收中斷
DMA_UR1T_STA = 0x00; //關閉dma 清零串口1DMA發送完成中斷標志、清零數據覆蓋中斷標志
DMA_UR1T_AMT = DMA_AMT_LEN; // 設置傳輸總字節數(低8位):n+1
DMA_UR1T_TXA = (u16)Dshow; // 設置傳輸數據的源地址
}
/*********************************************************
***********************************************************/
void DMA_Config(void)
{ DMA_TX(); DMA_RX(0);
}
/*********************************************************
***********************************************************/
void UART1_int (void) interrupt 4
{ if(RI) // 接收完成標志置1時
{ RI = 0; // 清零接收完成標志
if(Rx_cnt == DMA_AMT_LEN) TANG = 1; //就是這樣處理類似dma中斷的.郁悶啊!???
else {Rx_cnt++; TANG = 0; }
RX1_TimeOut = 5; // 如果 5ms 沒收到新的數據,判定一串數據接收完畢
}
if(TI) TI = 0; // 發送標志置1時
}
/*********************************************************
串口1初始化 (可用定時器1/2)
***********************************************************/
void UART1_Config(u8 brt)
{ u16 dat ;
dat = 65536UL- (MAIN_Fosc/4)/ Baudrate1;
if (brt==2)
{ AUXR|=0X01;
AUXR &=~(1<<4);
AUXR &=~(1<<3);
AUXR |=(1<<2);
T2H=(u8)(dat/256);
T2L=(u8)(dat%256);
IE2&= ~(1<<2);
AUXR |=(1<<4);
}
else
{ TR1=0;
AUXR&=~0X01;
AUXR|=(1<<6);
TMOD&=~(1<<6);
TMOD&=~0X30;
TH1=(u8)(dat/256);
TL1=(u8)(dat%256);
ET1=0;
INTCLKO&=0X02;
TR1=1;
}
SCON=(SCON&0X3F)|0X40;
//PS=1;
ES=1;
REN=1;
P_SW1&=0X3F;
P_SW1|=0X00;
}
/*********************************************************
***********************************************************/
void timer0(void) interrupt 1
{ B_1ms = 1; } // 1毫秒標志
/****************************************************
***************************************************/
void Timer0_Init(void) //1毫秒@22.1184MHz
{ AUXR |= 0x80; //定時器時鐘1T模式
TMOD &= 0xF0; //設置定時器模式
TMOD |= 0x01; //設置定時器模式
TL0 = 0x9A; //設置定時初始值
TH0 = 0xA9; //設置定時初始值
TF0 = 0; //清除TF0標志
ET0=1;
TR0 = 1; //定時器0開始計時
EA=1;
}
/***************************************************
***************************************************/
void GPIO_Config() //GPIO設置
{ P_SW2|=0X80;
P0M1 = 0x00;P0M0 = 0x00;
P1M1 = 0x00;P1M0 = 0x00;
P2M1 = 0x00;P2M0 = 0x00;
P3M1 = 0x00;P3M0 = 0x00;
P4M1 = 0x00;P4M0 = 0x00;
P5M1 = 0x00;P5M0 = 0x00;
}
/************************************************************
************************************************************/
void UartSendByte(u8 dat)
{ bit m=0;
if (ES){ES= 0;m=1;}
SBUF = dat;
while(TI == 0);
TI = 0;
ES= m; ;
}
/*********************************************************
***********************************************************/
void UartSendStr(u8 *str)
{ bit m=0;
if (ES){ES= 0;m=1;}
while(*str)
{ UartSendByte(*str);
str++;
} ES= m;
}
/***********************************************************
************************************************************
void UART1_DMAerrupt() interrupt 13 // 串口DMA中斷號大于31,借用13號保留中斷中轉
{ P46=0; // 詳情參照布丁橘長-STC32系列視頻第36期,或STC32手冊第5.9章節
if (DMA_UR1T_STA & 0x01) // 發送完成中斷標志為1時
{
DMA_UR1T_STA &= ~0x01; // 清零發送完成中斷標志
DMATxFlag = 1; // 發送完成標志置1
}
if (DMA_UR1T_STA & 0x04) // 數據覆蓋中斷標志為1時
{
DMA_UR1T_STA &= ~0x04; // 清零數據覆蓋中斷標志
}
if (DMA_UR1R_STA & 0x01) // 接收完成中斷標志為1時
{
DMA_UR1R_STA &= ~0x01; // 清零接收完成中斷標志
DMARxFlag = 1; // 接收完成標志置1
}
if (DMA_UR1R_STA & 0x02) // 數據丟棄中斷標志為1時
{
DMA_UR1R_STA &= ~0x02; // 清零數據丟棄中斷標志
}
}
/***********************************************************
************************************************************
if((DMATxFlag) && (DMARxFlag)) // 當發送和接收完成標志均為1時,表示空閑
{ Rx_cnt = 0; P45=0; // 清零接收計數
RX1_TimeOut = 0; // 清零接收超時計數
DMATxFlag = 0; // 清零發送完成標志
DMA_UR1R_STA = 0x00;
DMA_UR1T_AMT=Rx_cnt; //設置UART1傳輸總字節數:n+1
DMA_UR1T_TXA=Dshow; //設置UART1 DMA發送源地址
DMA_UR1T_CR = 0xc0; // bit7 1:使能 UART1_DMA, bit6 1:開始 UART1_DMA 自動發送
DMARxFlag = 0; // 清零接收完成標志
DMA_UR1R_CR = 0xa1; // bit7 1:使能 UART1_DMA, bit5 1:開始 UART1_DMA 自動接收, bit0 1:清除 FIFO
}
**********************************************************/
|