OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 6898|回复: 21

串口输出乱码问题 USART3 printf函数

[复制链接]

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
发表于 2018-9-6 14:16:34 | 显示全部楼层 |阅读模式
3金钱
之前用的F429VGT6USART功能进行串口输出,基本上没问题,但是最近用F429ZGT6,配置相同的程序出了问题,用不用DMA都是乱码,就是数据可以传输,但接收的是乱码,有人遇到这样的问题吗,原子的例程和我自己的都试过还不行
贴出代码
[mw_shl_code=c,true]#include "usart.h"
#include "delay.h"
//////////////////////////////////////////////////////////////////////////////////          
//如果使用os,则包括下面的头文件即可.
#if SYSTEM_SUPPORT_OS
#include "includes.h"                                        //os 使用          
#endif
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK STM32F429开发板
//串口1初始化                  
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//修改日期:2015/9/7
//版本:V1.5
//版权所有,盗版必究。
//Copyright(C) 广州市星翼电子科技有限公司 2009-2019
//All rights reserved
//********************************************************************************
//V1.0修改说明
//////////////////////////////////////////////////////////////////////////////////           
//加入以下代码,支持printf函数,而不需要选择use MicroLIB          
//#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)       
#if 1
#pragma import(__use_no_semihosting)            
//标准库需要的支持函数                 
struct __FILE
{
        int handle;
};

FILE __stdout;      
//定义_sys_exit()以避免使用半主机模式   
void _sys_exit(int x)
{
        x = x;
}
//重定义fputc函数
int fputc(int ch, FILE *f)
{        
        while((USART3->SR&0X40)==0);//循环发送,直到发送完毕   
        USART3->DR = (u8) ch;      
        return ch;
}
#endif

#if EN_USART3_RX   //如果使能了接收
//串口1中断服务程序
//注意,读取USARTx->SR能避免莫名其妙的错误          
u8 USART_RX_BUF[USART_REC_LEN];     //接收缓冲,最大USART_REC_LEN个字节.
//接收状态
//bit15,        接收完成标志
//bit14,        接收到0x0d
//bit13~0,        接收到的有效字节数目
u16 USART_RX_STA=0;       //接收状态标记       

u8 aRxBuffer[RXBUFFERSIZE];//HAL库使用的串口接收缓冲
UART_HandleTypeDef UART3_Handler; //UART句柄

//初始化IO 串口1
//bound:波特率
void uart_init(u32 bound)
{       
        //UART 初始化设置
        UART3_Handler.Instance=USART3;                                            //USART1
        UART3_Handler.Init.BaudRate=bound;                                    //波特率
        UART3_Handler.Init.WordLength=UART_WORDLENGTH_8B;   //字长为8位数据格式
        UART3_Handler.Init.StopBits=UART_STOPBITS_1;            //一个停止位
        UART3_Handler.Init.Parity=UART_PARITY_NONE;                    //无奇偶校验位
        UART3_Handler.Init.HwFlowCtl=UART_HWCONTROL_NONE;   //无硬件流控
        UART3_Handler.Init.Mode=UART_MODE_TX_RX;                    //收发模式
        HAL_UART_Init(&UART3_Handler);                                            //HAL_UART_Init()会使能UART1
       
        HAL_UART_Receive_IT(&UART3_Handler, (u8 *)aRxBuffer, RXBUFFERSIZE);//该函数会开启接收中断:标志位UART_IT_RXNE,并且设置接收缓冲以及接收缓冲接收最大数据量
  
}

//UART底层初始化,时钟使能,引脚配置,中断配置
//此函数会被HAL_UART_Init()调用
//huart:串口句柄

void HAL_UART_MspInit(UART_HandleTypeDef *huart)
{
    //GPIO端口设置
        GPIO_InitTypeDef GPIO_Initure;
       
        if(huart->Instance==USART3)//如果是串口1,进行串口1 MSP初始化
        {
                __HAL_RCC_GPIOD_CLK_ENABLE();                        //使能GPIOA时钟
                __HAL_RCC_USART3_CLK_ENABLE();                        //使能USART1时钟
       
                GPIO_Initure.Pin=GPIO_PIN_8;                        //PA9
                GPIO_Initure.Mode=GPIO_MODE_AF_PP;                //复用推挽输出
                GPIO_Initure.Pull=GPIO_PULLUP;                        //上拉
                GPIO_Initure.Speed=GPIO_SPEED_FAST;                //高速
                GPIO_Initure.Alternate=GPIO_AF7_USART3;        //复用为USART1
                HAL_GPIO_Init(GPIOD,&GPIO_Initure);                   //初始化PA9

                GPIO_Initure.Pin=GPIO_PIN_9;                        //PA10
                HAL_GPIO_Init(GPIOD,&GPIO_Initure);                   //初始化PA10
               
#if EN_USART3_RX
                HAL_NVIC_EnableIRQ(USART3_IRQn);                //使能USART1中断通道
                HAL_NVIC_SetPriority(USART3_IRQn,3,3);        //抢占优先级3,子优先级3
#endif       
        }

}

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
        if(huart->Instance==USART3)//如果是串口1
        {
                if((USART_RX_STA&0x8000)==0)//接收未完成
                {
                        if(USART_RX_STA&0x4000)//接收到了0x0d
                        {
                                if(aRxBuffer[0]!=0x0a)USART_RX_STA=0;//接收错误,重新开始
                                else USART_RX_STA|=0x8000;        //接收完成了
                        }
                        else //还没收到0X0D
                        {       
                                if(aRxBuffer[0]==0x0d)USART_RX_STA|=0x4000;
                                else
                                {
                                        USART_RX_BUF[USART_RX_STA&0X3FFF]=aRxBuffer[0] ;
                                        USART_RX_STA++;
                                        if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收          
                                }                 
                        }
                }

        }
}

//串口1中断服务程序
void USART3_IRQHandler(void)                       
{
        u32 timeout=0;
        u32 maxDelay=0x1FFFF;
#if SYSTEM_SUPPORT_OS                 //使用OS
        OSIntEnter();   
#endif
       
        HAL_UART_IRQHandler(&UART3_Handler);        //调用HAL库中断处理公用函数
       
        timeout=0;
    while (HAL_UART_GetState(&UART3_Handler) != HAL_UART_STATE_READY)//等待就绪
        {
         timeout++;////超时处理
     if(timeout>maxDelay) break;               
        }
     
        timeout=0;
        while(HAL_UART_Receive_IT(&UART3_Handler, (u8 *)aRxBuffer, RXBUFFERSIZE) != HAL_OK)//一次处理完成之后,重新开启中断并设置RxXferCount为1
        {
         timeout++; //超时处理
         if(timeout>maxDelay) break;       
        }
#if SYSTEM_SUPPORT_OS                 //使用OS
        OSIntExit();                                                                                           
#endif
}
#endif       


/*下面代码我们直接把中断控制逻辑写在中断服务函数内部。
说明:采用HAL库处理逻辑,效率不高。*/
/*


//串口1中断服务程序
void USART1_IRQHandler(void)                       
{
        u8 Res;
#if SYSTEM_SUPPORT_OS                 //使用OS
        OSIntEnter();   
#endif
        if((__HAL_UART_GET_FLAG(&UART1_Handler,UART_FLAG_RXNE)!=RESET))  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
        {
        HAL_UART_Receive(&UART1_Handler,&Res,1,1000);
                if((USART_RX_STA&0x8000)==0)//接收未完成
                {
                        if(USART_RX_STA&0x4000)//接收到了0x0d
                        {
                                if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
                                else USART_RX_STA|=0x8000;        //接收完成了
                        }
                        else //还没收到0X0D
                        {       
                                if(Res==0x0d)USART_RX_STA|=0x4000;
                                else
                                {
                                        USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
                                        USART_RX_STA++;
                                        if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收          
                                }                 
                        }
                }                    
        }
        HAL_UART_IRQHandler(&UART1_Handler);       
#if SYSTEM_SUPPORT_OS                 //使用OS
        OSIntExit();                                                                                           
#endif
}
#endif       
*/





[/mw_shl_code]

[mw_shl_code=c,true]#ifndef _USART_H
#define _USART_H
#include "sys.h"
#include "stdio.h"       
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK STM32F429开发板
//串口1初始化                  
//正点原子@ALIENTEK
//技术论坛:www.openedv.csom
//修改日期:2015/6/23
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) 正点原子 2009-2019
//All rights reserved
//********************************************************************************
//V1.0修改说明
//////////////////////////////////////////////////////////////////////////////////        
#define USART_REC_LEN                          200          //定义最大接收字节数 200
#define EN_USART3_RX                         1                //使能(1)/禁止(0)串口1接收
                 
extern u8  USART_RX_BUF[USART_REC_LEN]; //接收缓冲,最大USART_REC_LEN个字节.末字节为换行符
extern u16 USART_RX_STA;                         //接收状态标记       
extern UART_HandleTypeDef UART3_Handler; //UART句柄

#define RXBUFFERSIZE   1 //缓存大小
extern u8 aRxBuffer[RXBUFFERSIZE];//HAL库USART接收Buffer

//如果想串口中断接收,请不要注释以下宏定义
void uart_init(u32 bound);


#endif
[/mw_shl_code]


int main(void)
{
    u8 len;       
        u16 times=0;
    HAL_Init();                     //初始化HAL库   
    Stm32_Clock_Init(360,25,2,8);   //设置时钟,180Mhz
    delay_init(180);                //初始化延时函数
    uart_init(115200);              //初始化USART
//    LED_Init();                     //初始化LED
//    KEY_Init();                     //初始化按键

    while(1)
    {
               
                printf("123");
                delay_ms(10);

.......

最佳答案

查看完整内容[请看2#楼]

感谢大家的热情帮助,是接口弄错了,单片机直接输出的是TTL电平,我之前一直用的是232接口
我只喜欢喜欢我的人
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 14:16:35 | 显示全部楼层
感谢大家的热情帮助,是接口弄错了,单片机直接输出的是TTL电平,我之前一直用的是232接口
我只喜欢喜欢我的人
回复

使用道具 举报

5

主题

29

帖子

0

精华

初级会员

Rank: 2

积分
127
金钱
127
注册时间
2018-5-8
在线时间
32 小时
发表于 2018-9-6 14:17:47 | 显示全部楼层
看一下你的外部晶振有没有改
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 14:21:42 | 显示全部楼层
十年好唱却难等 发表于 2018-9-6 14:17
看一下你的外部晶振有没有改

没有改,晶振都是8MHz,我就在VGT6程序的基础上改了IO口,其他的VGT6和ZGT6也一样的啊,没什么改,但是数据乱码了
我只喜欢喜欢我的人
回复

使用道具 举报

5

主题

29

帖子

0

精华

初级会员

Rank: 2

积分
127
金钱
127
注册时间
2018-5-8
在线时间
32 小时
发表于 2018-9-6 14:24:34 | 显示全部楼层
正点原子的例程晶振不是8M吧?我没记错的话好像是25M
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 14:29:15 | 显示全部楼层
十年好唱却难等 发表于 2018-9-6 14:24
正点原子的例程晶振不是8M吧?我没记错的话好像是25M

我自己之前的VGT6是8M的,然后这个程序在ZGT6上也不行(已经匹配了IO口)
我只喜欢喜欢我的人
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 14:29:39 | 显示全部楼层
十年好唱却难等 发表于 2018-9-6 14:24
正点原子的例程晶振不是8M吧?我没记错的话好像是25M

应该不是晶振的问题吧
我只喜欢喜欢我的人
回复

使用道具 举报

2

主题

686

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
10887
金钱
10887
注册时间
2014-6-12
在线时间
1039 小时
发表于 2018-9-6 14:43:22 | 显示全部楼层
是不是单片机发送时,受到其他中断影响。
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 14:44:44 | 显示全部楼层
xuyaqi 发表于 2018-9-6 14:43
是不是单片机发送时,受到其他中断影响。

这个主函数我只开了一个串口,其他的外设都没用,就是循环发送延时
我只喜欢喜欢我的人
回复

使用道具 举报

21

主题

2205

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5141
金钱
5141
注册时间
2014-8-26
在线时间
1317 小时
发表于 2018-9-6 15:05:49 | 显示全部楼层
我们产品晶振是25M的,代码里面外部晶振的时钟宏定义都要改下
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 15:45:01 | 显示全部楼层
login_FAE 发表于 2018-9-6 15:05
我们产品晶振是25M的,代码里面外部晶振的时钟宏定义都要改下

    Stm32_Clock_Init(336,8,2,8);   //设置时钟,180Mhz
改过了好像还是一样的
我只喜欢喜欢我的人
回复

使用道具 举报

21

主题

2205

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5141
金钱
5141
注册时间
2014-8-26
在线时间
1317 小时
发表于 2018-9-6 16:11:57 | 显示全部楼层
fang百川 发表于 2018-9-6 15:45
Stm32_Clock_Init(336,8,2,8);   //设置时钟,180Mhz
改过了好像还是一样的

HSE_VALUE代码里面所有这个的宏定义都改的
回复

使用道具 举报

0

主题

113

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3321
金钱
3321
注册时间
2017-12-14
在线时间
406 小时
发表于 2018-9-6 16:16:57 | 显示全部楼层
HAL_UART_MspInit(); 这个初始化都没用开我很怀疑你是怎么收到数据的
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 16:36:12 | 显示全部楼层
feichangkunge 发表于 2018-9-6 16:16
HAL_UART_MspInit(); 这个初始化都没用开我很怀疑你是怎么收到数据的

uart_init(115200);              //初始化USART
这个不是初始化吗,初始化的代码我也贴在了一楼啊
我只喜欢喜欢我的人
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 16:54:53 | 显示全部楼层
login_FAE 发表于 2018-9-6 16:11
HSE_VALUE代码里面所有这个的宏定义都改的

我都改过了,
[mw_shl_code=c,true]#if !defined  (HSE_VALUE)
  #define HSE_VALUE    ((uint32_t)8000000) /*!< Value of the External oscillator in Hz */
#endif /* HSE_VALUE */
[/mw_shl_code]
有这个的宏定义我都改了,会不会是别的地方可能有错误
我只喜欢喜欢我的人
回复

使用道具 举报

21

主题

2205

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5141
金钱
5141
注册时间
2014-8-26
在线时间
1317 小时
发表于 2018-9-6 17:09:49 | 显示全部楼层
fang百川 发表于 2018-9-6 16:54
我都改过了,
[mw_shl_code=c,true]#if !defined  (HSE_VALUE)
  #define HSE_VALUE    ((uint32_t)80 ...

你板子用的8M晶振么?我们的蓝牙  GPS例程也是用的串口3的  可以看下
回复

使用道具 举报

5

主题

29

帖子

0

精华

初级会员

Rank: 2

积分
127
金钱
127
注册时间
2018-5-8
在线时间
32 小时
发表于 2018-9-6 17:19:54 | 显示全部楼层
延时太短了吧哈哈哈哈哈
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 17:21:50 | 显示全部楼层
login_FAE 发表于 2018-9-6 17:09
你板子用的8M晶振么?我们的蓝牙  GPS例程也是用的串口3的  可以看下

好的,谢谢热情帮助,我再多看看例程
我只喜欢喜欢我的人
回复

使用道具 举报

0

主题

113

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3321
金钱
3321
注册时间
2017-12-14
在线时间
406 小时
发表于 2018-9-6 17:25:17 | 显示全部楼层
fang百川 发表于 2018-9-6 16:36
uart_init(115200);              //初始化USART
这个不是初始化吗,初始化的代码我也贴在了一楼啊

这个是串口的初始化,HAL_UART_MspInit是IO口的初始化,你要在uart_init里面调用这个才行
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 17:45:39 | 显示全部楼层
feichangkunge 发表于 2018-9-6 17:25
这个是串口的初始化,HAL_UART_MspInit是IO口的初始化,你要在uart_init里面调用这个才行

我明白,你的意思是IO没有初始化对吧,但这个函数是回调函数,不用在uart里面写的(写了结果也是一样),而且原子的例程也是没写,我之前的程序结构都是这样的。个人见解
我只喜欢喜欢我的人
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-6 17:46:18 | 显示全部楼层
十年好唱却难等 发表于 2018-9-6 17:19
延时太短了吧哈哈哈哈哈

不清楚,我再看看吧
我只喜欢喜欢我的人
回复

使用道具 举报

27

主题

132

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
328
金钱
328
注册时间
2018-7-20
在线时间
117 小时
 楼主| 发表于 2018-9-7 08:59:44 | 显示全部楼层
再顶一下
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-9 06:42

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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