HAL库用UART发送数据产生pwm控制灯的亮度实验

最近花了两天时间,踩了大量的坑,总算是把hal库的串口收发数据相关的函数大概给弄明白了,把uart相关的函数逻辑设计的过于复杂可以说是hal库的一个问题,下面开始对hal库内有关串口通讯的函数内容做一个大致的解析

对于收发数据来说,假如不以dma方式进行收发,那么我们主要会接触到的函数有以下四个

HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);

HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);

HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
 

其中HAL_UART_Transmit与HAL_UART_Receive两个函数比较好理解,函数的定义也很详细,简单来说就是一定时间内完成发送,若设定的最大时间内仍然未完成则报一个溢出,我们在单片机内利用重映射来使用printf与scanf时,也是使用了这两个函数。这两个函数与中断无关。

在这里写一下printf和scanf的原型,平时大家可能会用得上。

int fputc(int ch, FILE *f)
{
     uint8_t ch;

      HAL_UART_Transmit(&huart1, (uint8_t *)&ch,1, 0xFFFF);
      
      return ch;
}
int fgetc(FILE *f)
{
      uint8_t  ch;
      
HAL_UART_Receive(&huart1,(uint8_t *)&ch, 1, 0xFFFF);
      
return  ch;
}

然后是重头戏,也是我花了两天去研究的程序逻辑:

HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

直接查看函数的定义基本上不会有任何收获,这一点也是我觉得hal库比较讨厌的地方,像这种逻辑比较跳跃的地方却没有多少注释。

我们来看一下执行一次串口中断的流程:

 (感谢正点原子的资料)

顺序是这样的UARTx_IRQHandler->HAL_UART_IRQHandler->UART_Receive_IT->HAL_UART_RxCpltCallback

你没有看错——UART_Receive_IT这个函数又是一个新的函数,不在我们上面讲的四个函数之列。这个函数的定义相对而言比较简单,大家自己可以很容易的看定义看懂。

然后关于HAL_UART_Receive_IT,它的一个工作流程,我理解成它是把我在函数入口处的输入传递给指针*pdata和RxXferCount,并且使能中断,这个中断以中断回调函数RxCpltCallback结尾,并且在中断结束之后就自动关闭中断(这点非常重要),因此一般会在while(1)或者中断回调函数中调用。

另外有一个问题,当我以这样的方式写程序时,数据的收发是失败的,我个人偏向于把这个问题的原因理解成通过Transmit这个函数之后,串口中断被结束了,导致整个中断无法再被使能,而把Transmit改成Transmit_IT后,这个问题就得到了解决。

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    HAL_UART_Transmit(&huart1,aRxbuffer,5,100);		
    HAL_UART_Receive_IT(&huart1,aRxbuffer,5);
	
}

贴一下我成功实现控制的代码:

unsigned char pwm_buff=0;
unsigned char aRxbuffer[2]={0};
uint8_t aTxbuffer[] = "*******开始启用串口*********\r\n";
uint8_t message[] = "the value of the light is:";

 
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration----------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_TIM2_Init();
	HAL_TIM_Base_Start_IT(&htim2);
	HAL_TIM_PWM_Start(&htim2,TIM_CHANNEL_2);
  /* USER CODE BEGIN 2 */
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
		HAL_UART_Receive_IT(&huart1,aRxbuffer,2);
		TIM2->CCR2 = 100-pwm_buff;
  /* USER CODE END WHILE */

  /* USER CODE BEGIN 3 */

  }
  /* USER CODE END 3 */

}

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
	if(huart->Instance == USART1)
	{
		pwm_buff=(aRxbuffer[0]-48)*10+(aRxbuffer[1]-48);
		HAL_UART_Transmit(&huart1,message,sizeof(message),100);
		HAL_UART_Transmit(&huart1,aRxbuffer,2,100);
		HAL_UART_Transmit(&huart1,"\r\n",2,100);
	}	
}

以及失败的代码:

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
      HAL_UART_Transmit_IT(&huart1,aRxbuffer,2);
      HAL_UART_Receive_IT(&huart1,aRxbuffer,2);
      counter=(aRxbuffer[0] - 0x30)*10 + (aRxbuffer[1] - 0x30);
      TIM2->CCR2 = counter;
}

匪夷所思的是下面这段代码是能够成功实现控制的 

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
		HAL_UART_Transmit_IT(&huart1,aRxbuffer,1);
                HAL_UART_Receive_IT(&huart1,aRxbuffer,1);
		if(aRxbuffer[0]=='1')
			counter=0;
		else if(aRxbuffer[0]=='0')
			counter=100;
		TIM2->CCR2 = counter;
}

目前还没有想明白到底问题出在哪里,有想法的同学可以在评论里留言

 

  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值