王昱顺 发表于 2025-3-6 13:09:00

marshallemon 发表于 2025-3-6 11:21
你的回复和你们另一个FAE的回复不一致,他的意思是DMA发送缓冲区数组和接收缓冲区数组是同一个缓冲区,不 ...

没有这个说法,接收和发送是可以指定不同的地址的,详见技术手册:


marshallemon 发表于 2025-3-6 15:14:29

神农鼎 发表于 2025-3-6 11:31
以我讲的为准,
指定发送缓冲区和接收缓冲区在2个不同的起始地址的缓冲区,



你说的这两个8K的缓冲区是MCU串口自带的缓冲区吧?不是人为定义的缓冲区?

神农鼎 发表于 2025-3-6 17:04:50

8A8K64D4, 总共只有一个共同的 8K XDATA;
在这同一个 8K XDATA中,指定1个区块做接收缓冲区;
在这同一个 8K XDATA中,指定1个区块做发送缓冲区;
2个缓冲区,都在一个共同的 8K XDATA 中;
如发生总线竞争,有硬件自动总线仲裁











marshallemon 发表于 2025-3-7 21:55:15

王昱顺 发表于 2025-3-3 22:36
这是因为DMA只能发送位于xdata部分的地址。
您程序中使用的ABuffer和BBuffer都是没有指定xdata的,所以他们 ...

我将那两个buffer定义到xdata区域还是不行,
到底啥情况啊,一个DMA让我折腾了好几天了

marshallemon 发表于 2025-3-7 21:56:16

神农鼎 发表于 2025-3-6 11:31
以我讲的为准,
指定发送缓冲区和接收缓冲区在2个不同的起始地址的缓冲区,



我将变量定义到xdata区域还是不行,你们这DMA有bug吧?我用了那么多MCU的DMA都没出现过类似这种问题

marshallemon 发表于 2025-3-7 21:56:38

/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCMCU.com --------------------------------------------*/
/* --- Web: www.STCMCUDATA.com---------------------------------------*/
/* --- QQ:800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序            */
/*---------------------------------------------------------------------*/

/*************功能说明    **************

串口1,2,3,4全双工中断方式收发通讯程序。

通过PC向MCU发送数据, MCU将收到的数据自动存入DMA空间.

当DMA空间存满设置大小的内容后,通过串口的DMA自动发送功能把存储空间的数据原样返回.

利用串口接收中断进行超时判断,超时没有收到新的数据,表示一串数据已经接收完毕.

用定时器做波特率发生器,建议使用1T模式(除非低波特率用12T),并选择可被波特率整除的时钟频率,以提高精度。

下载时, 选择时钟 22.1184MHz (用户可自行修改频率).

******************************************/

#include "stc8a8k64d4.h"
#include <string.h>
#include "stdio.h"

typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;

#define MAIN_Fosc       22118400L   //定义主时钟(精确计算115200波特率)
#define Baudrate1       115200L
#define Timer0_Reload   (65536UL -(MAIN_Fosc / 1000))       //Timer 0 中断频率, 1000次/秒

#define DMA_TX_LEN      255//设置传输总字节数:n+1
#define DMA_RX_LEN      255//设置传输总字节数:n+1

bit B_1ms;          //1ms标志
bit        DmaTx1Flag;
bit        DmaRx1Flag;

bit        Tx1SendFlag;
u8RX1_TimeOut;
u16 RX1_Cnt;    //接收计数

u8 xdata DmaBuffer1;

u8 xdata ABuffer1[]={1,2,3};
u8 xdata BBuffer1[]={3,2,1,0};

void UART1_config(u8 brt);   // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.

void UART1_DMA_Config(void);


void UART1_DMA_Transmit(u8 *pData, u16 Size);
void UART1_DMA_Receive(u8 *pData, u16 Size);

void UartPutc(unsigned char dat)
{
    Tx1SendFlag = 1;
        SBUF = dat;
        while(Tx1SendFlag);
}

char putchar(char c)
{
        UartPutc(c);
        return c;
}

//========================================================================
// 函数: void main(void)
// 描述: 主函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void main(void)
{
        u16 i;
        u8 bflag=0;
       
        P_SW2 |= 0x80;   //扩展寄存器(XFR)访问使能

        P0M1 = 0x00;   P0M0 = 0x00;   //设置为准双向口
        P1M1 = 0x00;   P1M0 = 0x00;   //设置为准双向口
        P2M1 = 0x00;   P2M0 = 0x00;   //设置为准双向口
        P3M1 = 0x00;   P3M0 = 0x00;   //设置为准双向口
        P4M1 = 0x00;   P4M0 = 0x00;   //设置为准双向口
        P5M1 = 0x20;   P5M0 = 0x00;   //设置为准双向口, 设置P5.5(蜂鸣器)高阻输入
        P6M1 = 0x00;   P6M0 = 0x00;   //设置为准双向口
        P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口

        for(i=0; i<256; i++)
        {
                DmaBuffer1 = i;
        }

        AUXR = 0x80;    //Timer0 set as 1T, 16 bits timer auto-reload,
        TH0 = (u8)(Timer0_Reload / 256);
        TL0 = (u8)(Timer0_Reload % 256);
        ET0 = 1;    //Timer0 interrupt enable
        TR0 = 1;    //Tiner0 run
       
        UART1_config(1);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.

        UART1_DMA_Config();

        EA = 1; //允许总中断

        DmaTx1Flag = 0;
        DmaRx1Flag = 0;


        printf("STC8A8K64D4 UART DMA Test Programme!\r\n");//UART1发送一个字符串
        while (1)
        {
                if(B_1ms)   //1ms到
                {
                        B_1ms = 0;
                        if(RX1_TimeOut > 0)   //超时计数
                        {
                                if(--RX1_TimeOut == 0)
                                {
                                        //关闭接收DMA,下次接收的数据重新存放在起始地址位置,否则下次接收数据继续往后面存放。
                                        DMA_UR1R_CR = 0x00;                        //bit7 1:使能 UART1_DMA, bit5 1:开始 UART1_DMA 自动接收, bit0 1:清除 FIFO
                                        printf("\r\nUART1 Timeout!\r\n");//UART1发送一个字符串
                                        if(bflag)
                                        {
                                                ABuffer1=1;
                                                ABuffer1=2;
                                                ABuffer1=3;                                               
                                                UART1_DMA_Transmit(ABuffer1,sizeof(ABuffer1));
                                        }
                                        else
                                        {
                                                BBuffer1=3;
                                                BBuffer1=2;
                                                BBuffer1=1;
                                                BBuffer1=0;
                                                UART1_DMA_Transmit(BBuffer1,sizeof(BBuffer1));
                                        }                                       
                                        bflag=!bflag;
                  RX1_Cnt = 0;
                  DmaTx1Flag = 0;
                  DmaRx1Flag = 0;
                  UART1_DMA_Receive(DmaBuffer1,256);//设置DMA接收缓冲区,数据长度,并启动接收
                                }
                        }
                }

                if((DmaTx1Flag) && (DmaRx1Flag))
                {
                        RX1_TimeOut = 0;
                        printf("\r\nUART1 DMA FULL!\r\n");                //UART1发送一个字符串
//                        UART1_DMA_Transmit(DmaBuffer1,RX1_Cnt);        //设置DMA发送缓冲区,数据长度,并启动发送
                        RX1_Cnt = 0;
                        DmaTx1Flag = 0;
                        DmaRx1Flag = 0;
                        UART1_DMA_Receive(DmaBuffer1,256);                //设置DMA接收缓冲区,数据长度,并启动接收
//                        DmaTx1Flag = 0;
//                        DMA_UR1T_CR = 0xc0;                        //bit7 1:使能 UART1_DMA, bit6 1:开始 UART1_DMA 自动发送
//                        DmaRx1Flag = 0;
//                        DMA_UR1R_CR = 0xa1;                        //bit7 1:使能 UART1_DMA, bit5 1:开始 UART1_DMA 自动接收, bit0 1:清除 FIFO
                }
        }
}

//========================================================================
// 函数: void UART1_DMA_Config(void)
// 描述: UART1 DMA 功能配置.
// 参数: none.
// 返回: none.
// 版本: V1.0, 2021-5-6
//========================================================================
void UART1_DMA_Config(void)
{
        DMA_UR1T_CFG = 0x80;                //bit7 1:Enable Interrupt
        DMA_UR1T_STA = 0x00;
        DMA_UR1T_AMT = sizeof(BBuffer1);                //设置传输总字节数:n+1
        DMA_UR1T_TXA = BBuffer1;
//        DMA_UR1T_CR = 0xc0;                        //bit7 1:使能 UART1_DMA, bit6 1:开始 UART1_DMA 自动发送
        DMA_UR1T_CR = 0x80;                        //bit7 1:使能 UART1_DMA, bit6 0:非开始 UART1_DMA 自动发送

        DMA_UR1R_CFG = 0x80;                        //bit7 1:Enable Interrupt
        DMA_UR1R_STA = 0x00;
        DMA_UR1R_AMT = DMA_RX_LEN;                //设置传输总字节数:n+1
        DMA_UR1R_RXA = DmaBuffer1;
        DMA_UR1R_CR = 0xa1;                                //bit7 1:使能 UART1_DMA, bit5 1:开始 UART1_DMA 自动接收, bit0 1:清除 FIFO
}

void UART1_DMA_Transmit(u8 *pData, u16 Size)
{
    if(Size == 0) return;
    Size -= 1;
        DMA_UR1T_CFG = 0x80;                //bit7 1:Enable Interrupt
        DMA_UR1T_STA = 0x00;
        DMA_UR1T_AMT = (u8)Size;      //设置传输总字节数:n+1
        DMA_UR1T_TXA = (u8)pData;
    DMA_UR1T_CR = 0xc0;                   //bit7 1:使能 UART4_DMA, bit6 1:开始 UART1_DMA 自动发送
}

void UART1_DMA_Receive(u8 *pData, u16 Size)
{
    if(Size == 0) return;
    Size -= 1;
        DMA_UR1R_AMT = (u8)Size;      //设置传输总字节数:n+1
        DMA_UR1R_RXA = (u8)pData;
    DMA_UR1R_CR = 0xa1;   //bit7 1:使能 UART4_DMA, bit5 1:开始 UART4_DMA 自动接收, bit0 1:清除 FIFO
}

//========================================================================
// 函数: SetTimer2Baudraye(u16 dat)
// 描述: 设置Timer2做波特率发生器。
// 参数: dat: Timer2的重装值.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void SetTimer2Baudraye(u16 dat)// 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
        AUXR &= ~(1<<4);    //Timer stop
        AUXR &= ~(1<<3);    //Timer2 set As Timer
        AUXR |=(1<<2);    //Timer2 set as 1T mode
        T2H = dat / 256;
        T2L = dat % 256;
        IE2&= ~(1<<2);    //禁止中断
        AUXR |=(1<<4);    //Timer run enable
}

//========================================================================
// 函数: void UART1_config(u8 brt)
// 描述: UART1初始化函数。
// 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_config(u8 brt)    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
        /*********** 波特率使用定时器2 *****************/
        if(brt == 2)
        {
                AUXR |= 0x01;       //S1 BRT Use Timer2;
                SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate1);
        }

        /*********** 波特率使用定时器1 *****************/
        else
        {
                TR1 = 0;
                AUXR &= ~0x01;      //S1 BRT Use Timer1;
                AUXR |=(1<<6);    //Timer1 set as 1T mode
                TMOD &= ~(1<<6);    //Timer1 set As Timer
                TMOD &= ~0x30;      //Timer1_16bitAutoReload;
                TH1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) / 256);
                TL1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) % 256);
                ET1 = 0;    //禁止中断
                TR1= 1;
        }
        /*************************************************/

        SCON = (SCON & 0x3f) | 0x40;    //UART1模式, 0x00: 同步移位输出, 0x40: 8位数据,可变波特率, 0x80: 9位数据,固定波特率, 0xc0: 9位数据,可变波特率
//PS= 1;    //高优先级中断
        ES= 1;    //允许中断
        REN = 1;    //允许接收
        P_SW1 &= 0x3f;
        P_SW1 |= 0x40;      //UART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7, 0xC0: P4.3 P4.4

        RX1_TimeOut = 0;
}

//========================================================================
// 函数: void UART1_int (void) interrupt UART1_VECTOR
// 描述: UART1中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_int (void) interrupt 4
{
        if(RI)
        {
                RI = 0;
      RX1_Cnt++;
                RX1_TimeOut = 5;    //如果5ms没收到新的数据,判定一串数据接收完毕
        }

        if(TI)
        {
                TI = 0;
      Tx1SendFlag = 0;
        }
}

//========================================================================
// 函数: void timer0 (void) interrupt 1
// 描述: 定时器0中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void timer0 (void) interrupt 1
{
    B_1ms = 1;      //1ms标志
}

//========================================================================
// 函数: void UART_DMA_Interrupt (void)
// 描述: UART DMA中断函数
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2021-5-8
// 备注:
//========================================================================
void UART_DMA_Interrupt(void) interrupt 13
{
        //======= UART1 DMA ================
        if (DMA_UR1T_STA & 0x01)        //发送完成
        {
                DMA_UR1T_STA &= ~0x01;
                DmaTx1Flag = 1;
        }
        if (DMA_UR1T_STA & 0x04)        //数据覆盖
        {
                DMA_UR1T_STA &= ~0x04;
        }
       
        if (DMA_UR1R_STA & 0x01)        //接收完成
        {
                DMA_UR1R_STA &= ~0x01;
                DmaRx1Flag = 1;
        }
        if (DMA_UR1R_STA & 0x02)        //数据丢弃
        {
                DMA_UR1R_STA &= ~0x02;
        }
}

marshallemon 发表于 2025-3-7 22:07:08

我都比较奇怪,这DMA发送的是哪的内容?尝试了多种方法,测试了多次,结果都是一致的,我就奇怪了,这是哪里来的数据?

王昱顺 发表于 2025-3-8 00:07:13

marshallemon 发表于 2025-3-7 22:07
我都比较奇怪,这DMA发送的是哪的内容?尝试了多种方法,测试了多次,结果都是一致的,我就奇怪了,这是哪 ...
DMA_UR1T_TXA = (u8)pData;这一句是有问题的
这个地址寄存器是16位的地址寄存器,例程里是直接给的数据地址
所以这里的pData强制转换为8位后会裁断高八位的地址。
应该写作DMA_UR1T_TXA = pData;就没问题了
另外,数据手册中,提供了串口dma相关例程,都是调通的代码,可以参考测试

手机刷机菜鸟 发表于 2025-3-8 06:50:30

这个有意思讲的不错!

手机刷机菜鸟 发表于 2025-3-8 06:50:33

这个有意思讲的不错!
页: 1 [2] 3
查看完整版本: 8A8K64D4 串口DMA实现乒乓发送问题