STM32G0开发笔记-Platformio+libopencm3-串口中断

【原文】https://makerinchina.cn/article_dfa2d9ee8875.html


title: STM32G0开发笔记-Platformio+libopencm3-串口中断
tags:

  • STM32
  • MCU
  • STM32G070
  • libopencm3
  • MonkeyPi
    categories:
    • STM32
      date: 2022-9-5 09:33:05

使用Platformio平台的libopencm3开发框架来开发STM32G0,以下为串口中断的使用。

1 新建项目
  • 建立uart项目

在PIO的Home页面新建项目,项目名称uart,选择开发板为上一次建立的自定义开发板型号 MonkeyPi_STM32_G070RB,开发框架选择libopencm3;

  • 项目建立完成后在src目录下新建main.c主程序文件;
  • 修改下载和调试方式,这里开发板使用的是DAPLink仿真器,因此修改platformio.ini文件如下:
upload_protocol = cmsis-dap
debug_tool = cmsis-dap
2 编写程序
  • 时钟设置
//system clock
rcc_clock_setup(&rcc_clock_config[RCC_CLOCK_CONFIG_HSI_PLL_64MHZ]);

//uart pin
rcc_periph_clock_enable(RCC_USART1);
rcc_periph_clock_enable(RCC_GPIOB);

先设置系统时钟为内部PLL生成64MHz,然后设置串口外设和串口引脚外设的时钟;

  • 引脚复用功能设置
gpio_mode_setup(GPIOB,GPIO_MODE_AF,GPIO_PUPD_NONE,GPIO6|GPIO7);
gpio_set_af(GPIOB,GPIO_AF0,GPIO6|GPIO7);

根据芯片datasheet文档,使用USART1其引脚为PA9\PA10,复用功能AF1为串口功能;

  • 串口设置
usart_set_baudrate(USART1,115200);
usart_set_databits(USART1,8);
usart_set_stopbits(USART1,USART_STOPBITS_1);
usart_set_parity(USART1,USART_PARITY_NONE);
usart_set_flow_control(USART1,USART_FLOWCONTROL_NONE);
usart_set_mode(USART1,USART_MODE_TX_RX);

//uart isr
nvic_enable_irq(NVIC_USART1_IRQ);

usart_enable(USART1);

usart_enable_rx_interrupt(USART1);

先设置串口波特率、数据位数、停止位、校验、流控等设置,再开启串口中断,使能串口和其接收中断;

  • 串口发送
char buff[32] = "hello, makerinchina.cn\n";
for(int i=0; i<strlen(buff); i++){
    usart_send_blocking(USART1, buff[i]);
}

发送直接使用 usart_send_blocking 接口发送一个字节数据;

  • 串口接收
/**
 * @brief uart1 isr function
 * 
 */
void usart1_isr(void)
{
    //receive interrupt
	if (((USART_CR1(USART1) & USART_CR1_RXNEIE) != 0) &&
	    ((USART_ISR(USART1) & USART_ISR_RXNE) != 0)) {

        if(recv_index < BUFF_SIZE){
            recv_buff[recv_index++] = usart_recv(USART1);
        }else{
            recv_index = 0;
        }
	}
}

串口接收使用中断方式,接收到数据后将其存放的buff中,然后主程序中取出打印显示出来:

#include <libopencm3/stm32/usart.h>
#include <libopencm3/stm32/rcc.h>
#include <libopencm3/stm32/gpio.h>
#include <libopencm3/cm3/nvic.h>

#include <string.h>

volatile uint8_t recv_index = 0;
volatile uint8_t send_index = 0;

#define BUFF_SIZE 64
uint8_t recv_buff[BUFF_SIZE] = {0};

int main(void)
{
    ...
        
    while (1)
    {

        if(recv_index != send_index){ 

            if(send_index < BUFF_SIZE){
                usart_send_blocking(USART1, recv_buff[send_index++]);
            }else{
                send_index = 0;
            }
        }

    }
}

注:recv_index、send_index需要声明为volatile类型;

3 烧写测试

点击 platformio:Upload按钮或快捷键 Ctrl+Alt+U即可完成编译、烧写过程,打开串口工具,可以看到发送的数据和接收数据一样;

4 printf使用

如果要使用printf功能,需要定义如下函数,将串口重定向:

/*
 * Called by libc stdio fwrite functions
 */
int
_write(int fd, char *ptr, int len)
{
	int i = 0;

	/*
	 * Write "len" of char from "ptr" to file id "fd"
	 * Return number of char written.
	 *
	 * Only work for STDOUT, STDIN, and STDERR
	 */
	if (fd > 2) {
		return -1;
	}
	while (*ptr && (i < len)) {
		usart_send_blocking(USART1, *ptr);
		if (*ptr == '\n') {
			usart_send_blocking(USART1, '\r');
		}
		i++;
		ptr++;
	}
	return i;
}

现在就可以直接使用printf函数进行打印显示到串口设备;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
针对STM32G4HAL库串口空闲中断接收不定长数据的问题,可以按照以下步骤进行解决: 1. 开启串口空闲中断,即在初始化串口时,将USART_CR1_REG中的USART_CR1_IDLEIE位设置为1。 2. 在串口空闲中断中,通过读取USART_ISR_REG寄存器中的位USART_ISR_RXNE和USART_ISR_IDLE来判断是接收到数据还是空闲中断。 3. 如果是接收到数据,则读取USART_RDR_REG寄存器中的数据,并将其存储到缓冲区中。 4. 如果是空闲中断,则通过计算接收到的数据长度来确定数据的长度,并将其存储到缓冲区中。 5. 在数据长度达到预定长度时,可以通过回调函数或者其他方式来通知数据接收已经完成。 下面是一个示例代码: ```c uint8_t rx_buffer[100]; uint8_t rx_counter = 0; uint8_t rx_length = 0; uint8_t rx_flag = 0; void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart->Instance == USARTx) { if(rx_flag == 0) { //接收到数据 if((__HAL_UART_GET_FLAG(huart, UART_FLAG_RXNE) != RESET) && (__HAL_UART_GET_FLAG(huart, UART_FLAG_IDLE) == RESET)) { rx_buffer[rx_counter++] = (uint8_t)(huart->Instance->RDR & 0x00FF); } //空闲中断 else if(__HAL_UART_GET_FLAG(huart, UART_FLAG_IDLE) != RESET) { rx_length = rx_counter; rx_counter = 0; rx_flag = 1; } } } } ``` 在上面的代码中,当接收到数据时,将数据存储到缓冲区中,并将计数器rx_counter加1。当空闲中断发生时,计算接收到的数据长度,并将其存储到rx_length中。在接收完成后,将rx_flag设置为1,表示数据接收已经完成。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值