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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2339|回復: 3
打印 上一主題 下一主題
收起左側

使用stc8h8k64u單片機的DMA處理串口數據

[復制鏈接]
跳轉到指定樓層
樓主
ID:73833 發表于 2023-10-30 14:43 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
/*******************************************************************
串口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
        }
         
        **********************************************************/
分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:213173 發表于 2023-10-30 15:32 | 只看該作者
新版STC8H手冊1009頁有范例
回復

使用道具 舉報

板凳
ID:216265 發表于 2023-12-10 10:52 | 只看該作者
stm32 那么強,那么好用,你還用STC干嘛?
回復

使用道具 舉報

地板
ID:195496 發表于 2023-12-11 08:28 | 只看該作者
沒有串口沒有空閑中斷嗎?
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: av片在线免费看 | 国产女人叫床高潮大片免费 | 国产亚洲欧美另类一区二区三区 | 九色在线视频 | 久久国产精品久久久久久久久久 | a中文在线视频 | 亚州春色| 久久国产精品网 | av免费观看网站 | 成人免费淫片aa视频免费 | 午夜激情视频 | 999久久| 在线观看国产视频 | 中文字幕在线网 | 中文字幕免费中文 | 美人の美乳で授乳プレイ | 91在线视频免费观看 | 亚洲社区在线 | 久久久女 | 免费毛片网站在线观看 | 国产69精品久久久久777 | 久久久久久久久久久蜜桃 | 亚洲看片网站 | 欧美一级高清片 | av看看 | 日韩天堂av| 国产一级视频在线播放 | 一级片免费视频 | 天天射色综合 | 天天拍天天操 | 69堂永久69tangcom | 午夜看片网站 | 亚洲国产精品99久久久久久久久 | 成人毛片视频免费 | 国产乱码精品一区二区三区忘忧草 | 性色av香蕉一区二区 | 日韩在线播放中文字幕 | 久久久久久久久99精品 | 欧美日韩一区二区三区四区 | 国产在线视频一区二区董小宛性色 | 操操操av |