[MM32硬件] 【灵动微电子MM32F0121测评】2、串口输入输出及呼吸灯

[复制链接]
 楼主| 穿西装的强子 发表于 2025-6-13 21:17 | 显示全部楼层 |阅读模式
本帖最后由 穿西装的强子 于 2025-6-13 22:32 编辑

1、串口功能
参考例程USART_Interrupt例程进行移植由于该例程是usart2的,需要改为usart1。
28263684c24e661c59.png

将plartfrom.c和mm32f0120_it.c内容进行移植到我们工程内
90809684c25269a476.png
plartfrom主要是串口输出功能,已经指向了printf,不需要自己再额外的修改了
唯一需要注意的是需要添加Use MicroLib功能
26994684c25788e7ff.png
移植好了以后打印看输出结果
成功打印出“Hello 21ic, HelloMM32F0121”,还有主频等信息
只需要在主函数起始调用PLATFORM_Init();函数即可实现串口配置功能。
78838684c25d99c28d.png
但是该串口只有打印没有接收功能,现在先将接收也移植进来。
在void PLATFORM_InitConsole(uint32_t Baudrate)函数内添加中断接收及接收引脚的配置。
配置代码如下
  1. void PLATFORM_InitConsole(uint32_t Baudrate)
  2. {
  3.     GPIO_InitTypeDef GPIO_InitStruct;
  4.     NVIC_InitTypeDef  NVIC_InitStruct;
  5.     USART_InitTypeDef USART_InitStruct;

  6.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);

  7.     USART_StructInit(&USART_InitStruct);
  8.     USART_InitStruct.USART_BaudRate      = Baudrate;
  9.     USART_InitStruct.USART_WordLength    = USART_WordLength_8b;
  10.     USART_InitStruct.USART_StopBits      = USART_StopBits_1;
  11.     USART_InitStruct.USART_Parity        = USART_Parity_No;
  12.     USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  13.     USART_InitStruct.USART_Mode          = USART_Mode_Rx | USART_Mode_Tx;
  14.     USART_Init(USART1, &USART_InitStruct);

  15.     RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOB, ENABLE);

  16.     GPIO_PinAFConfig(GPIOB, GPIO_PinSource6, GPIO_AF_0);
  17.     GPIO_PinAFConfig(GPIOB, GPIO_PinSource7, GPIO_AF_0);

  18.     GPIO_StructInit(&GPIO_InitStruct);
  19.     GPIO_InitStruct.GPIO_Pin   = GPIO_Pin_6;
  20.     GPIO_InitStruct.GPIO_Speed = GPIO_Speed_High;
  21.     GPIO_InitStruct.GPIO_Mode  = GPIO_Mode_AF_PP;
  22.     GPIO_Init(GPIOB, &GPIO_InitStruct);
  23.        
  24.     GPIO_StructInit(&GPIO_InitStruct);
  25.     GPIO_InitStruct.GPIO_Pin   = GPIO_Pin_7;
  26.     GPIO_InitStruct.GPIO_Mode  = GPIO_Mode_FLOATING;
  27.     GPIO_Init(GPIOB, &GPIO_InitStruct);
  28.        
  29.     NVIC_InitStruct.NVIC_IRQChannel = USART1_IRQn;
  30.     NVIC_InitStruct.NVIC_IRQChannelPriority = 0;
  31.     NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
  32.     NVIC_Init(&NVIC_InitStruct);

  33.     USART_ITConfig(USART1, USART_IT_PE, ENABLE);
  34.     USART_ITConfig(USART1, USART_IT_ERR, ENABLE);

  35.     USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

  36.     USART_Cmd(USART1, ENABLE);
  37. }
串口中断增加接收函数,并通过串口回传(暂时只做串口回发)
  1. void USART1_IRQHandler(void)
  2. {
  3.     uint8_t RxData = 0;

  4.         if ((RESET != USART_GetITStatus(USART1, USART_IT_PE)) ||
  5.         (RESET != USART_GetITStatus(USART1, USART_IT_ERR)))
  6.     {
  7.         USART_ReceiveData(USART1);
  8.     }

  9.     if (RESET != USART_GetITStatus(USART1, USART_IT_RXNE))
  10.     {
  11.         RxData = USART_ReceiveData(USART1);

  12.                 USART_SendData(USART1, RxData);
  13.         
  14.     }

  15. }
串口打印和接收
39269684c2e697190c.png
2、PWM呼吸灯效果
根据原理图LED在PB14和PB15上
2666684c2ea3c0406.png
该引脚使用TIM1_CH1/CH3和TIM1_CH2两个引脚接了三个通道,我们只需要使用2个通道即可,就使用CH1和CH2
1913684c2ee9c46a8.png
使用固件库的例程代码TIM1_PWM_Output进行移植
85521684c33f7a934f.png
由于TIM1_PWM_Output代码内使用的是PB3/PB3/PB5,和LED引脚不匹配,因此需要修改改引脚,再将三通道改为两通道
再根据用户手册可知道,PB14和PB15使用TIM1的配置是AP7的功能,因此需要将
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource3,  GPIO_AF_6);   /* TIM1_CH1 */
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource4,  GPIO_AF_6);   /* TIM1_CH2 */
改为
  GPIO_PinAFConfig(GPIOB, GPIO_PinSource14,  GPIO_AF_7);   /* TIM1_CH1 */
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource15,  GPIO_AF_7);   /* TIM1_CH2 */

GPIO的输出也配置为GPIO14和GPIO15
删除TIM_OC3Init的配置
78902684c34405801b.png
代码如下
  1. RCC_ClocksTypeDef       RCC_Clocks;
  2.     GPIO_InitTypeDef        GPIO_InitStruct;
  3.     TIM_OCInitTypeDef       TIM_OCInitStruct;
  4.     TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  5.     uint32_t                TIM_ClockFrequency = 0;
  6.     uint32_t                HPRE = 0, PPRE2 = 0;

  7.     uint32_t TimerPeriod = 0, Channel1Pulse = 0, Channel2Pulse = 0;

  8.     HPRE  = READ_BIT(RCC->CFGR, RCC_CFGR_HPRE)  >> RCC_CFGR_HPRE_Pos;
  9.     PPRE2 = READ_BIT(RCC->CFGR, RCC_CFGR_PPRE2) >> RCC_CFGR_PPRE2_Pos;

  10.     RCC_GetClocksFreq(&RCC_Clocks);

  11.     if (HPRE < 8)
  12.     {
  13.         if (PPRE2 < 4)
  14.         {
  15.             TIM_ClockFrequency = RCC_Clocks.PCLK1_Frequency;
  16.         }
  17.         else
  18.         {
  19.             TIM_ClockFrequency = RCC_Clocks.PCLK1_Frequency * 2;
  20.         }
  21.     }
  22.     else
  23.     {
  24.         if (PPRE2 < 4)
  25.         {
  26.             TIM_ClockFrequency = RCC_Clocks.PCLK1_Frequency * 2;
  27.         }
  28.         else
  29.         {
  30.             TIM_ClockFrequency = RCC_Clocks.PCLK1_Frequency * 4;
  31.         }
  32.     }

  33.     /* Compute the value to be set in ARR regiter to generate signal frequency at 100 Khz */
  34.     TimerPeriod = TIM_ClockFrequency / 100000;

  35.     /* Compute CCR1 value to generate a duty cycle at 75% for channel 1 */
  36.     Channel1Pulse = (uint32_t)1000 * TimerPeriod / 1000;

  37.     /* Compute CCR2 value to generate a duty cycle at 50% for channel 2 */
  38.     Channel2Pulse = (uint32_t)0 * TimerPeriod / 1000;


  39.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);

  40.     TIM_TimeBaseStructInit(&TIM_TimeBaseInitStruct);
  41.     TIM_TimeBaseInitStruct.TIM_Prescaler         = 0;
  42.     TIM_TimeBaseInitStruct.TIM_CounterMode       = TIM_CounterMode_Up;
  43.     TIM_TimeBaseInitStruct.TIM_Period            = TimerPeriod;
  44.     TIM_TimeBaseInitStruct.TIM_ClockDivision     = TIM_CKD_Div1;
  45.     TIM_TimeBaseInitStruct.TIM_RepetitionCounter = 0;
  46.     TIM_TimeBaseInit(TIM1, &TIM_TimeBaseInitStruct);

  47.     TIM_OCStructInit(&TIM_OCInitStruct);
  48.     TIM_OCInitStruct.TIM_OCMode       = TIM_OCMode_PWM1;
  49.     TIM_OCInitStruct.TIM_OutputState  = TIM_OutputState_Enable;
  50.     TIM_OCInitStruct.TIM_Pulse        = 0;
  51.     TIM_OCInitStruct.TIM_OCPolarity   = TIM_OCPolarity_High;
  52.     TIM_OCInitStruct.TIM_OCIdleState  = TIM_OCIdleState_Set;

  53.     TIM_OCInitStruct.TIM_Pulse = Channel1Pulse;
  54.     TIM_OC1Init(TIM1, &TIM_OCInitStruct);

  55.     TIM_OCInitStruct.TIM_Pulse = Channel2Pulse;
  56.     TIM_OC2Init(TIM1, &TIM_OCInitStruct);


  57.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB, ENABLE);

  58.     GPIO_PinAFConfig(GPIOB, GPIO_PinSource14,  GPIO_AF_7);   /* TIM1_CH1 */
  59.     GPIO_PinAFConfig(GPIOB, GPIO_PinSource15,  GPIO_AF_7);   /* TIM1_CH2 */

  60.     GPIO_StructInit(&GPIO_InitStruct);
  61.     GPIO_InitStruct.GPIO_Pin   = GPIO_Pin_14 | GPIO_Pin_15 ;
  62.     GPIO_InitStruct.GPIO_Speed = GPIO_Speed_High;
  63.     GPIO_InitStruct.GPIO_Mode  = GPIO_Mode_AF_PP;
  64.     GPIO_Init(GPIOB, &GPIO_InitStruct);

  65.     TIM_Cmd(TIM1, ENABLE);

  66.     TIM_CtrlPWMOutputs(TIM1, ENABLE);
在主函数控制LED的呼吸效果
  1. int main(void)
  2. {
  3.         uint32_t i = 0;
  4.         PLATFORM_Init();
  5. //        GPIO_LED_Toggle_Sample();
  6.         TIM1_PWM_Output_Sample();
  7.         printf("\r\nHello 21ic, HelloMM32F0121");
  8.        
  9.     while (1)
  10.     {
  11.                 TIM1->CCR1 = 1440 - (i++%1440);
  12.                 TIM1->CCR2 = (i++%1440);

  13.         PLATFORM_DelayMS(2);

  14.     }
  15. }
视频效果如下


22744684c24dbe0013.png
您需要登录后才可以回帖 登录 | 注册

本版积分规则

62

主题

259

帖子

3

粉丝
快速回复 在线客服 返回列表 返回顶部