久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

標題: 使用stc8h8k64u單片機的DMA處理串口數據 [打印本頁]

作者: dingmingzhou    時間: 2023-10-30 14:43
標題: 使用stc8h8k64u單片機的DMA處理串口數據
/*******************************************************************
串口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
        }
         
        **********************************************************/

作者: wulin    時間: 2023-10-30 15:32
新版STC8H手冊1009頁有范例
作者: dcc024    時間: 2023-12-10 10:52
stm32 那么強,那么好用,你還用STC干嘛?
作者: 大漠落日    時間: 2023-12-11 08:28
沒有串口沒有空閑中斷嗎?




歡迎光臨 (http://m.zg4o1577.cn/bbs/) Powered by Discuz! X3.1
主站蜘蛛池模板: 欧美精品一区二区在线观看 | 亚洲成人免费视频 | 国产成年妇视频 | 国产一区视频在线播放 | 黄色一级片网站 | av片在线观看 | 午夜激情视频在线观看 | 男女搞黄网站 | 国产三级在线观看视频 | 国产在线一区二区三区 | 国产精品毛片av | 黄色片www | 黄色av大片 | 亚洲黄色网址 | 一级毛片黄色 | 少妇视频网站 | 亚洲综合视频在线观看 | 亚洲日本在线观看 | 久本草精品| 欧美性猛交xxxx | 日日操av | 国产一区二区不卡视频 | 午夜激情福利视频 | 一级免费片 | 国产精品久久久久久久久久久久午夜片 | 天天色小说 | 99re在线观看视频 | 日韩av在线免费播放 | 日本中文字幕在线播放 | 国产视频导航 | 天天干天天草 | 国产理论在线 | 欧美在线免费观看视频 | 欧美在线视频一区二区 | 中文字幕日韩视频 | 成av人片在线观看www | 久久机热这里只有精品 | 日本免费观看视频 | www.com国产| 91在线免费视频观看 | 久久久久久久97 |