上位机图像处理和嵌入式模块部署(f103 mcu中的串口接口)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】

        在mcu开发中,mcu扮演着非常重要的角色。一方面,串口可以帮助我们对固件功能进行调试,另外一方面,串口还是很好的模块通信工具。通常情况下,一个mcu里面会有若干个串口,不过因为mcu里面各个pin脚都是复用的,所以这几个串口能不能全部用起来,取决于对应的pin脚有没有被占领。

        除了串口之外,其他的接口也有类似的功能,比如说spi、i2c、sdio、485、can、eth等等。这些都是接口的形式,但是接口上面的数据以什么样的协议进行数据传输,这就要具体的模块或者是芯片了。用一个比喻来说,接口相当于搭了一条路,数据好比路上面的车,这过程中一次发送多少车,都是不定的。以i2c为例,它对应的模块可能是一个陀螺仪芯片,也有可能是一个eeprom的芯片,所以协议这部分要查看具体的芯片手册才能做决定。

        今天,我们回头看下,stm32f103c8t6是怎么实现串口的输入、输出的。

1、从main函数看起

        拿到示例工程后,我们一般直接编译一下。编译除了判断是否成功之外,编译日志还会告诉我们工程中使用的rom多少、ram多少,这也是十分重要的,

Program Size: Code=4516 RO-data=360 RW-data=20 ZI-data=1092  

        解析来简单看下main函数内容,

int main(void)
{
    HAL_Init();        
    SystemClock_Config();
    DEBUG_USART_Config();
  
	printf("Welcome to use fire board!\r\n");	
	Usart_SendString( (uint8_t *)"This is just an echo experiment\r\n" );
	
    while(1)
	{		
	}
}

        整个函数中比较明显的区别就是DEBUG_USART_Config函数,其他的就是打印和字符输出。最后while循环是空的,这代表对应的处理都是在中断完成的。

2、串口初始化

void DEBUG_USART_Config(void)
{ 
  
  UartHandle.Instance          = DEBUG_USART;
  
  UartHandle.Init.BaudRate     = DEBUG_USART_BAUDRATE;
  UartHandle.Init.WordLength   = UART_WORDLENGTH_8B;
  UartHandle.Init.StopBits     = UART_STOPBITS_1;
  UartHandle.Init.Parity       = UART_PARITY_NONE;
  UartHandle.Init.HwFlowCtl    = UART_HWCONTROL_NONE;
  UartHandle.Init.Mode         = UART_MODE_TX_RX;
  
  HAL_UART_Init(&UartHandle);
  __HAL_UART_ENABLE_IT(&UartHandle,UART_IT_RXNE);  
}

void HAL_UART_MspInit(UART_HandleTypeDef *huart)
{  
  GPIO_InitTypeDef  GPIO_InitStruct;
  
  DEBUG_USART_CLK_ENABLE();
	
  DEBUG_USART_RX_GPIO_CLK_ENABLE();
  DEBUG_USART_TX_GPIO_CLK_ENABLE();
  
/**USART1 GPIO Configuration    
  PA9     ------> USART1_TX
  PA10    ------> USART1_RX 
  */
	
  GPIO_InitStruct.Pin = DEBUG_USART_TX_PIN;
  GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed =  GPIO_SPEED_FREQ_HIGH;
  HAL_GPIO_Init(DEBUG_USART_TX_GPIO_PORT, &GPIO_InitStruct);
  
  GPIO_InitStruct.Pin = DEBUG_USART_RX_PIN;
  GPIO_InitStruct.Mode=GPIO_MODE_AF_INPUT;	
  HAL_GPIO_Init(DEBUG_USART_RX_GPIO_PORT, &GPIO_InitStruct); 
 
  HAL_NVIC_SetPriority(DEBUG_USART_IRQ ,0,1);
  HAL_NVIC_EnableIRQ(DEBUG_USART_IRQ ); 
}

        DEBUG_USART_Config函数中主要就是对串口的一些属性进行配置,比如波特率多少、数据长度多少、停止位多少等等。这里就是常规的115200-8-1配置。配置完了,会调用函数HAL_UART_Init,这个函数会进一步调用下面的HAL_UART_MspInit。而在HAL_UART_MspInit函数当中,主要就是对a9、a10两个pin脚进行配置,配置完了设置中断优先级,并且使能中断。而在DEBUG_USART_Config函数的最后一步,则是全局配置中断,也就是接收数据的时候才响应中断。

3、补充fputc和fgetc函数

        我们自己写代码的时候,习惯用printf和getchar这样的函数。如果需要这样的函数,就需要显示开发一下fputc和fgetc。这两个函数都是WEAK函数。所谓的WEAK函数,就是工程里面已经有一个函数了,但是客户如果自己重新定义了,那就用客户自己定义的函数。

void Usart_SendString(uint8_t *str)
{
	unsigned int k=0;
    do 
    {
      HAL_UART_Transmit(&UartHandle,(uint8_t *)(str + k) ,1,1000);
      k++;
    } while(*(str + k)!='\0'); 
}

int fputc(int ch, FILE *f)
{
	HAL_UART_Transmit(&UartHandle, (uint8_t *)&ch, 1, 1000);	
	
	return (ch);
}

int fgetc(FILE *f)
{		
	int ch;
	HAL_UART_Receive(&UartHandle, (uint8_t *)&ch, 1, 1000);	
	return (ch);
}

4、中断响应

        在stm32 mcu里面,主要的中断都是在文件stm32f1xx_it.c里面完成的。所以我们需要做的,就是直接找到对应的内容,看看是怎么实现的即可,

void  DEBUG_USART_IRQHandler(void)
{
    uint8_t ch=0; 
  
	if(__HAL_UART_GET_FLAG( &UartHandle, UART_FLAG_RXNE ) != RESET)
	{		
        ch=( uint16_t)READ_REG(UartHandle.Instance->DR);
        WRITE_REG(UartHandle.Instance->DR,ch); 
	}
}

        __HAL_UART_GET_FLAG主要是查看一下是不是真的有数据,READ_REG是读取数据,WRITE_REG是发送数据。所以整个代码的功能就是一个回显的功能。

5、实验和测试

        要进行实际验证的话,除了下载好固件之外,还需要进行串口连接,打开mobaxterm。这时候如果可以回显对应的内容,那么就说明固件没问题。不然就会回头check一下硬件、接线和软件了。

6、其他

        因为是涉及到串口,一般的pc电脑已经没有串口的接口,所以需要提前购买一个usb转232的模块。另外,我们实践过程中,一般是把调试串口和协议串口是分开来处理的。

        串口本身的意义还是十分巨大的,它使得我们通过外部设备,以协议的形式对这个模块进行控制了。之前我们学习了gpio的读和写,那么就可以通过串口实现gpio的读、写控制,这样的话就可以编写上位机对mcu模块进行控制了。这个上位机可以是pc,可以是嵌入式linux。

        mcu固件开发中,大部分应用都是事件触发式的,上位机给一个命令,我们处理一个命令。如果是需要定时反馈的,这个时候就再添加一个定时器就可以了。

  • 20
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式-老费

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值