OpenEdv-开源电子网

 找回密码
 立即注册

扫一扫,访问微社区

正点原子新作:阿波罗STM32F767&F429&探索者STM32F4开发板&赶快来下载资料哦。

查看: 112|回复: 6

使用f103的芯片,用usart1接485,和电脑通信只能发送不能接收

[复制链接]

  离线 

1

主题

3

帖子

0

精华

新手上路

Rank: 1

积分
15
金钱
15
注册时间
2017-11-14
在线时间
0 小时
发表于 2017-11-14 10:39:17 | 显示全部楼层 |阅读模式
1金钱
int main(void)
{               
        u8 i=0,t=0;
        u8 cnt=0;
        u8 rec_rs485buf[9];
        u8 sed_rs485buf[8]={0x01,0x03,0x00,0x00,0x00,0x02,0xC4,0x0B};
       
        Stm32_Clock_Init(6);        //系统时钟设置
        uart_init(72,9600);                 //串口初始化为9600
        delay_init(72);                            //延时初始化                 
                                                                  
        while(1)
        {
                RS485_Send_Data(sed_rs485buf,8);//发送8个字节                                                                           
                delay_ms(1000);         
                RS485_Receive_Data(rec_rs485buf,&key);
        }
}
******************************************************************************
//加入以下代码,支持printf函数,而不需要选择use MicroLIB          
#if 1
#pragma import(__use_no_semihosting)            
//标准库需要的支持函数                 
struct __FILE
{
        int handle;
        /* Whatever you require here. If the only file you are using is */
        /* standard output using printf() for debugging, no file handling */
        /* is required. */
};
/* FILE is typedef’ d in stdio.h. */
FILE __stdout;      
//定义_sys_exit()以避免使用半主机模式   
_sys_exit(int x)
{
        x = x;
}
//重定义fputc函数
int fputc(int ch, FILE *f)
{      
        while((USART2->SR&0X40)==0);//循环发送,直到发送完毕   
        USART2->DR = (u8) ch;      
        return ch;
}
#endif
//end
//////////////////////////////////////////////////////////////////

#if EN_USART1_RX   //如果使能了接收
//串口1中断服务程序
//注意,读取USARTx->SR能避免莫名其妙的错误          
u8 USART_RX_BUF[USART_REC_LEN];     //接收缓冲,最大USART_REC_LEN个字节.

u16 USART_RX_STA=0;       //接收状态标记          

void USART1_IRQHandler(void)
{
        u8 res;       
#ifdef OS_CRITICAL_METHOD         //如果OS_CRITICAL_METHOD定义了,说明使用ucosII了.
        OSIntEnter();   
#endif
        if(USART1->SR&(1<<5))//接收到数据
        {
                res=USART1->DR;                          
                if(USART_RX_STA<64)
                {
                        USART_RX_BUF[USART_RX_STA]=res;                //记录接收到的值
                        USART_RX_STA++;                                                //接收数据增加1
                }                                                                                               
        }
#ifdef OS_CRITICAL_METHOD         //如果OS_CRITICAL_METHOD定义了,说明使用ucosII了.
        OSIntExit();                                                                                           
#endif
}
#endif                                                                                 
//初始化IO 串口1
//pclk2CLK2时钟频率(Mhz)
//bound:波特率
//CHECK OK
//091209
void uart_init(u32 pclk2,u32 bound)
{           
        float temp;
        u16 mantissa;
        u16 fraction;          
        temp=(float)(pclk2*1000000)/(bound*16);//得到USARTDIV
        mantissa=temp;                                 //得到整数部分
        fraction=(temp-mantissa)*16; //得到小数部分         
        mantissa<<=4;
        mantissa+=fraction;
       
        RCC->APB2ENR|=1<<8;           //使能PORTG9口时钟   485使能端
        GPIOG->CRH&=0XFFFFFF0F;        //IO状态设置
        GPIOG->CRH|=0X00000030;        //IO状态设置
       
        RCC->APB2ENR|=1<<2;   //使能PORTA口时钟  
        RCC->APB2ENR|=1<<14;  //使能串口时钟
        GPIOA->CRH&=0XFFFFF00F;//IO状态设置
        GPIOA->CRH|=0X000008B0;//IO状态设置
                  
        RCC->APB2RSTR|=1<<14;   //复位串口1
        RCC->APB2RSTR&=~(1<<14);//停止复位                     
        //波特率设置
        USART1->BRR=mantissa; // 波特率设置         
        USART1->CR1|=0X200C;  //1位停止,无校验位.
#if EN_USART1_RX                  //如果使能了接收
        //使能接收中断
        USART1->CR1|=1<<8;    //PE中断使能
        USART1->CR1|=1<<5;    //接收缓冲区非空中断使能                   
        MY_NVIC_Init(3,3,USART1_IRQn,2);//组2,最低优先级
#endif
        RS485_TX_EN=0;                        //默认为接收模式       
}


//RS485发送len个字节.
//buf:发送区首地址
//len:发送的字节数(为了和本代码的接收匹配,这里建议不要超过64个字节)
void RS485_Send_Data(u8 *buf,u8 len)
{
        u8 t;
        RS485_TX_EN=1;                        //设置为发送模式
          for(t=0;t<len;t++)                //循环发送数据
        {
                while((USART1->SR&0X40)==0);//等待发送结束                  
                USART1->DR=buf[t];
        }         
        while((USART1->SR&0X40)==0);//等待发送结束       
        USART_RX_STA=0;          
        RS485_TX_EN=0;                                //设置为接收模式       
}
//RS485查询接收到的数据
//buf:接收缓存首地址
//len:读到的数据长度
void RS485_Receive_Data(u8 *buf,u8 *len)
{
        u8 rxlen=USART_RX_STA;
        u8 i=0;
        *len=0;                                //默认为0
        delay_ms(10);                //等待10ms,连续超过10ms没有接收到一个数据,则认为接收结束
        if(rxlen==USART_RX_STA&&rxlen)//接收到了数据,且接收完成了
        {
                for(i=0;i<rxlen;i++)
                {
                        buf[i]=USART_RX_BUF[i];       
                }               
                *len=USART_RX_STA;        //记录本次数据长度
                USART_RX_STA=0;                //清零
        }
}
*********************************************************************************************
#ifndef __USART_H
#define __USART_H
#include "sys.h"
#include "stdio.h"         

#define USART_REC_LEN                          200          //定义最大接收字节数 200
#define EN_USART1_RX                         1                //使能(1)/禁止(0)串口1接收

#define RS485_TX_EN                PGout(9)                //485模式控制.0,接收;1,发送.
                 
extern u8  USART_RX_BUF[USART_REC_LEN]; //接收缓冲,最大USART_REC_LEN个字节.末字节为换行符
extern u16 USART_RX_STA;                         //接收状态标记       
//如果想串口中断接收,请不要注释以下宏定义
void uart_init(u32 pclk2,u32 bound);

void RS485_Send_Data(u8 *buf,u8 len);
void RS485_Receive_Data(u8 *buf,u8 *len);

#endif          



回复

使用道具 举报

  离线 

1

主题

3

帖子

0

精华

新手上路

Rank: 1

积分
15
金钱
15
注册时间
2017-11-14
在线时间
0 小时
 楼主| 发表于 2017-11-14 10:57:14 | 显示全部楼层
回复

使用道具 举报

  离线 

1

主题

3

帖子

0

精华

新手上路

Rank: 1

积分
15
金钱
15
注册时间
2017-11-14
在线时间
0 小时
 楼主| 发表于 2017-11-14 13:59:05 | 显示全部楼层
求帮忙啊
回复

使用道具 举报

  离线 

482

主题

8万

帖子

30

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
117247
金钱
117247
注册时间
2010-12-1
在线时间
900 小时
发表于 2017-11-14 20:14:43 | 显示全部楼层
仿真找下问题
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
微信公众平台:正点原子   点击扫码添加
回复

使用道具 举报

  离线 

5

主题

25

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
471
金钱
471
注册时间
2015-5-19
在线时间
43 小时
发表于 2017-11-14 20:59:30 | 显示全部楼层
RS485_Receive_Data()函数里面的rxlen==USART_RX_STA&&rxlen判断,当没接收到数据和接收到的数据长度为1时,一直通过;当接收到的数据长度大于1时,不通过。是不是这里有问题?
回复

使用道具 举报

  离线 

18

主题

118

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
243
金钱
243
注册时间
2015-10-16
在线时间
49 小时
发表于 7 天前 | 显示全部楼层
在中断处理函数USART1_IRQHandler()里设断点,看看能进去不
回复

使用道具 举报

  离线 

9

主题

29

帖子

0

精华

新手上路

Rank: 1

积分
42
金钱
42
注册时间
2017-11-2
在线时间
5 小时
发表于 7 天前 | 显示全部楼层
中断开了么?
回复

使用道具 举报

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

本版积分规则




QQ|联系我们|手机版|官方淘宝店|新浪微博|微信公众平台|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2017-11-22 07:11

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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