菜鸟大神 发表于 2025-3-5 21:01:25

官方例程里面串口1使用DMA中断模式与电脑收发测试

程序是可以正常运行的,但是DMA初始化的时候会出现一段乱码,是什么原因

例程源码如下
/*---------------------------------------------------------------------*/
/* --- 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.STCAI.com ---------------------------------------------*/
/* --- Web: www.STCMCUDATA.com---------------------------------------*/
/* --- BBS: www.STCAIMCU.com-----------------------------------------*/
/* --- QQ:800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序            */
/*---------------------------------------------------------------------*/

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

本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8H系列带DMA模块的芯片可通用参考.

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

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

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

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

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

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

#include "..\..\comm\STC8H.h"
#include "stdio.h"
#include "intrins.h"

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

#define MAIN_Fosc       24000000L   //定义主时钟(精确计算115200波特率)
#define Baudrate1       115200L

bit        DmaTxFlag;
bit        DmaRxFlag;

u8 xdata DmaBuffer;

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

void UartPutc(unsigned char dat)
{
        SBUF = dat;
        while(TI == 0);
        TI = 0;
}

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

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

    P0M1 = 0x30;   P0M0 = 0x30;   //设置P0.4、P0.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P1M1 = 0x30;   P1M0 = 0x30;   //设置P1.4、P1.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P2M1 = 0x3c;   P2M0 = 0x3c;   //设置P2.2~P2.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P3M1 = 0x50;   P3M0 = 0x50;   //设置P3.4、P3.6为漏极开路(实验箱加了上拉电阻到3.3V)
    P4M1 = 0x3c;   P4M0 = 0x3c;   //设置P4.2~P4.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P5M1 = 0x0c;   P5M0 = 0x0c;   //设置P5.2、P5.3为漏极开路(实验箱加了上拉电阻到3.3V)
    P6M1 = 0xff;   P6M0 = 0xff;   //设置为漏极开路(实验箱加了上拉电阻到3.3V)
    P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口

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

        UART1_config(1);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
        printf("STC8H UART1 DMA Test Programme!\r\n");//UART1发送一个字符串

        DMA_Config();
        EA = 1; //允许总中断

        DmaTxFlag = 0;
        DmaRxFlag = 0;

        while (1)
        {
                if((DmaTxFlag) && (DmaRxFlag))
                {
                        DmaTxFlag = 0;
                        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
                }
        }
}

//========================================================================
// 函数: void DMA_Config(void)
// 描述: UART DMA 功能配置.
// 参数: none.
// 返回: none.
// 版本: V1.0, 2021-5-6
//========================================================================
void DMA_Config(void)
{
        DMA_UR1T_CFG = 0x80;                //bit7 1:Enable Interrupt
        DMA_UR1T_STA = 0x00;
        DMA_UR1T_AMT = 0xff;                //设置传输总字节数:n+1
        DMA_UR1T_TXAH = (u8)((u16)&DmaBuffer >> 8);
        DMA_UR1T_TXAL = (u8)((u16)&DmaBuffer);
        DMA_UR1T_CR = 0xc0;                        //bit7 1:使能 UART1_DMA, bit6 1:开始 UART1_DMA 自动发送

        DMA_UR1R_CFG = 0x80;                //bit7 1:Enable Interrupt
        DMA_UR1R_STA = 0x00;
        DMA_UR1R_AMT = 0xff;                //设置传输总字节数:n+1
        DMA_UR1R_RXAH = (u8)((u16)&DmaBuffer >> 8);
        DMA_UR1R_RXAL = (u8)((u16)&DmaBuffer);
        DMA_UR1R_CR = 0xa1;                        //bit7 1:使能 UART1_DMA, bit5 1:开始 UART1_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((u16)(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;    //禁止中断
                INTCLKO &= ~0x02;//不输出时钟
                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 |= 0x00;      //UART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7, 0xC0: P4.3 P4.4
}


//========================================================================
// 函数: 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_Buffer = SBUF;
//                if(++RX1_Cnt >= UART1_BUF_LENGTH)   RX1_Cnt = 0;    //防溢出
//        }

//        if(TI)
//        {
//                TI = 0;
//                B_TX1_Busy = 0;
//        }
//}

//========================================================================
// 函数: void UART1_DMA_Interrupt (void) interrupt 50/51
// 描述: UART1 DMA中断函数
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2021-5-8
// 备注:
//========================================================================
void UART1_DMA_Interrupt(void) interrupt 13
{
        if (DMA_UR1T_STA & 0x01)        //发送完成
        {
                DMA_UR1T_STA &= ~0x01;
                DmaTxFlag = 1;
        }
        if (DMA_UR1T_STA & 0x04)        //数据覆盖
        {
                DMA_UR1T_STA &= ~0x04;
        }
       
        if (DMA_UR1R_STA & 0x01)        //接收完成
        {
                DMA_UR1R_STA &= ~0x01;
                DmaRxFlag = 1;
        }
        if (DMA_UR1R_STA & 0x02)        //数据丢弃
        {
                DMA_UR1R_STA &= ~0x02;
        }
}
然后串口输出如下。串口助手是STCISP6.95版本,使用P30 31输出数据,COM口硬件是STC-USB LINK1D
接收←.
接收←
接收←郤TC8H UART1 DMA Test Programme!
                  .........       
                  .................... !"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~€亗儎厗噲墛媽崕彁憭摂晼棙櫄洔潪煚ⅲぅΗī氨渤吹斗腹夯冀究懒旅呐魄壬仕掏蜗醒矣哉肿刭谯茌捱噌忏溴骁栝觌祉铒瘃蝮趱鲼?纯好奇为啥会这样,是程序问题还是硬件问题

国芯人工智能 发表于 2025-3-5 21:23:00

关于串口1使用DMA中断模式时初始化出现乱码的问题,可能的原因包括以下几个方面:

1. **DMA缓冲区未正确初始化**:在DMA初始化之前,缓冲区可能未清零或未正确配置,导致读取到随机数据。建议在DMA配置前对缓冲区进行清零操作。

2. **波特率不匹配**:串口通信的波特率设置与电脑端不一致,可能导致数据解析错误。请确保双方波特率设置一致。

3. **DMA传输长度设置错误**:DMA传输长度可能未正确配置,导致传输数据超出预期范围。请检查DMA传输长度参数。

4. **时钟源不稳定**:MCU时钟源不稳定或配置错误,可能导致串口通信异常。建议检查时钟配置。

建议逐一排查以上问题,确保DMA和串口配置正确。如有必要,可参考相关硬件手册进行深入调试。

[本文内容由国芯人工智能辅助生成,仅供参考]

国芯人工智能 发表于 2025-3-5 21:28:00

在DMA初始化过程中出现乱码,可能与以下几个原因有关:首先,检查DMA配置是否正确,特别是缓冲区地址和长度是否匹配。其次,确保时钟配置和波特率设置无误,避免因时钟不同步导致的数据错乱。此外,硬件连接问题,如线路干扰或接触不良,也可能引发乱码。建议逐步排查,先确认软件配置无误,再检查硬件连接,确保系统稳定运行。

[本文内容由国芯人工智能辅助生成,仅供参考]

DebugLab 发表于 2025-3-5 21:40:22

初始化时先用memset清空DmaBuffer试试
memset(DmaBuffer,0x00,sizeof DmaBuffer);

菜鸟大神 发表于 2025-3-6 20:19:25

DebugLab 发表于 2025-3-5 21:40
初始化时先用memset清空DmaBuffer试试
memset(DmaBuffer,0x00,sizeof DmaBuffer);

51里面没有memset。我写了一个// 自定义 memset 函数,用于将指定内存区域设置为指定值
void my_memset(void *s, unsigned char c, unsigned int n) {
    unsigned char *p = (unsigned char *)s;
    while (n--) {
      *p++ = c;
    }
}然后现在没有乱码了 但是变成了一排小点点
TC8H UART1 DMA Test Programme!
                  ................................................................................................................................................................................................................................................................{:ciya:}

jiangshan1xiao 发表于 2025-3-7 15:52:30

u8 xdata DmaBuffer;这个DmaBuffer是不是必须是xdata空间的?

乘风飞扬 发表于 2025-3-7 18:42:47

jiangshan1xiao 发表于 2025-3-7 15:52
u8 xdata DmaBuffer;这个DmaBuffer是不是必须是xdata空间的?

做DMA缓冲区使用的话,必需用 xdata 空间
页: [1]
查看完整版本: 官方例程里面串口1使用DMA中断模式与电脑收发测试