找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1523|回复: 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
        }
         
        **********************************************************/
回复

使用道具 举报

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 单片机教程网

快速回复 返回顶部 返回列表