OpenEdv-开源电子网

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

USART数据中断发送接收求助 新手 谢谢

[复制链接]

1

主题

2

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2018-10-27
在线时间
0 小时
发表于 2018-10-27 11:04:33 | 显示全部楼层 |阅读模式
1金钱
大家好 ,新手刚开始学习 搞了半天没有摸索出来 还望指点一下
1. 串口通讯 中断接收OK
2. 在时钟中断里面 超时1秒把收到的数据发出出去 ,这里直接 发送出去 while等待发送完毕 也OK
3. 现在想实现用中断发送 每次发送1BYTE 中断里面发送完毕继续发送下一BYTE 直到发完为止  没有成功

目前的流程在时钟中断里面超时后先发送1BYTE ,剩余的在中断里面发送 发现只能发送1BYTE ,不会进发送中断.



代码如下

串口初始化
void USART_Config(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;

    /* 使能 USART1 时钟*/
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);

    /* USART1 使用IO端口配置 */
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOA, &GPIO_InitStructure);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;        //浮空输入
    GPIO_Init(GPIOA, &GPIO_InitStructure);   //初始化GPIOA

    /* USART1 工作模式配置 */
    USART_InitStructure.USART_BaudRate = 115200;        //波特率设置:115200
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;        //数据位数设置:8位
    USART_InitStructure.USART_StopBits = USART_StopBits_1;         //停止位设置:1位
    USART_InitStructure.USART_Parity = USART_Parity_No ;  //是否奇偶校验:无
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;        //硬件流控制模式设置:没有使能
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//接收与发送都使能
               
    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); //允许串口1接收中断。。
                USART_ITConfig(USART1, USART_IT_TXE,   ENABLE); //允许串口1发送中断。。
               
    USART_Init(USART1, &USART_InitStructure);          //初始化USART1
    USART_Cmd(USART1, ENABLE);// USART1使能
}



串口中断
void nvic_USART1_init()
{
    // 串口一中断初始化
    NVIC_InitTypeDef nvic;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
    nvic.NVIC_IRQChannelPreemptionPriority = 0;
    nvic.NVIC_IRQChannelSubPriority = 0;
    nvic.NVIC_IRQChannel = USART1_IRQn;
    nvic.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&nvic);
}



void USART1_IRQHandler(void)
{
    // 串口1中断
    if(USART_GetFlagStatus(USART1, USART_IT_RXNE) != RESET) //判断是否有接收中断
    {
        // 保存uart数据到 uart_receive_buff 每次读取1BYTE
        uart_timeout_1000ms = 0;        // 接收到数据包 超时复位
        uart_rx_buffer[uart_rx_len]  = USART_ReceiveData(USART1);
        uart_rx_len++;
        if ( uart_rx_len >= 64)
        {
            uart_rx_len = 0;
        }
        USART_ClearITPendingBit(USART1, USART_IT_RXNE);         //清除接收中断
    }

    /*当USART数据发送完成后进入中断*/
    if(USART_GetFlagStatus(USART1, USART_IT_TXE) != RESET) //判断是否有发送中断
    {
        // 接收完毕 发送给PC
        if ( uart_rx_len > 0 )
        {
            UARTSendByte(uart_rx_buffer[uart_tx_index]);
            uart_tx_index ++;
            uart_rx_len --;
        }
        else
        {
            uart_tx_task        = 0;
            uart_rx_len         = 0;
            uart_tx_index = 0;
            USART_ITConfig(USART1, USART_IT_TXE, DISABLE);         //关闭发送中断
        }
    }
}


时钟中断
void TIM3_IRQHandler()
{
    // STM32中必须按照要求写中断函数名才能进入中断
    u8 i;
    if(TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)
    {
        uart_timeout_1000ms++;
        if(uart_timeout_1000ms >= 1000) //实现1秒一计数
        {
            // 接收完毕 发送给PC
            if ( uart_rx_len > 0 && 0 == uart_tx_task)
            {
                uart_tx_task = 1;// 开始发送
                uart_tx_index = 0;
                USART_SendData(USART1, uart_rx_buffer[uart_tx_index]);
                uart_rx_len --;
                uart_tx_index ++;
                if ( uart_rx_len == 0 )
                {
                    uart_tx_task = 0;
                    uart_tx_index = 0;
                }
                USART_ITConfig(USART1, USART_IT_TXE, ENABLE);  //这里立即启动了发送中断
            }
        }
    }
    TIM_ClearITPendingBit(TIM3, TIM_FLAG_Update); //清除中断标志位避免重复进入中断
}




正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

57

主题

1680

帖子

3

精华

资深版主

Rank: 8Rank: 8

积分
4307
金钱
4307
注册时间
2018-6-30
在线时间
808 小时
发表于 2018-10-27 11:45:45 | 显示全部楼层
这个仿真找下问题,期待早日解决,分享压缩版
业精于勤荒于嬉;行成于思毁于随!
回复

使用道具 举报

14

主题

70

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
395
金钱
395
注册时间
2018-7-16
在线时间
52 小时
发表于 2018-10-27 14:32:38 | 显示全部楼层
建议你把定时器中断实现的发送数据的功能写在主函数的while里面,中断服务程序要尽可能地简单,只是简单地计时就可以了。
回复

使用道具 举报

1

主题

2

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2018-10-27
在线时间
0 小时
 楼主| 发表于 2018-10-27 15:41:19 | 显示全部楼层
sunsetrain 发表于 2018-10-27 14:32
建议你把定时器中断实现的发送数据的功能写在主函数的while里面,中断服务程序要尽可能地简单,只是简单地 ...

谢谢 我来仿真一下 第一次调试STM32 还没有入门
回复

使用道具 举报

14

主题

70

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
395
金钱
395
注册时间
2018-7-16
在线时间
52 小时
发表于 2018-10-27 16:47:55 | 显示全部楼层
atkken 发表于 2018-10-27 15:41
谢谢 我来仿真一下 第一次调试STM32 还没有入门

加油  调试的时候就可以发现问题了
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-6-21 17:27

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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