HAL STM32G4 +TIM1 3路PWM互补输出+VOFA波形演示

2024-03-23 09:28

本文主要是介绍HAL STM32G4 +TIM1 3路PWM互补输出+VOFA波形演示,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

HAL STM32G4 +TIM1 3路PWM互补输出+VOFA波形演示


✨最近学习研究无刷电机驱动,虽然之前有使用过,但是在STM32上还没实现过。本文内容参考欧拉电子例程,从PWM驱动开始学习。

  • 欧拉电子相关视频讲解:

STM32G4 FOC开发实战—高级定时器发波

  • ✨本篇重点学习,如何配置PWM互补输出,和死区时间计算和设定,如果看了本篇还不会配置和计算死区时间,可以提着99米砍刀来找我。😁
  • 📍有关VOFA数据传输协议可以参考《VOFA+上位机三种协议(FireWater,JustFloat,RawData)C语言参考代码》

🛠STM32CubeMX工程配置

  • 🌿时钟配置:
    在这里插入图片描述

  • 🌿定时器1(TIM1)配置:

在这里插入图片描述

  • 🌿计数值:8000、计数方式:采用中心对称方式、时钟分频系数:2
    在这里插入图片描述
  • 🌿死区时间配置:(TIMx timer deadtime register 2 (TIMx_DTR2)(x = 1, 8, 20))
  • Dead Time:120,其值对应的二进制值:B0111 1000
    在这里插入图片描述
DTGF[7:5] = 0xx => DTF = DTGF[7:0]x tdtg with tdtg = tDTS.

在这里插入图片描述

    • 🔖死区时间换算:120/160MHz/2=1.5us
  • ✨死区时间参数,需要根据具体使用的MOS型号参数来调整。

  • 🔖逻辑分析仪死区时间测量:
    在这里插入图片描述

  • 🌾如果上的Dead Time配置为160,其对应的二进制值:B1010 0000
DTGF[7:5] = 10x => DTF = (64+DTGF[5:0])xtdtg with Tdtg = 2xtDTS
    • 🔖Dead Time配置为160,对应的死区时间换算:2*(64+32)/(160MHz/2)=2.4us
      在这里插入图片描述
  • 🌾如果上的Dead Time配置为200,其对应的二进制值:B1100 1000:
DTGF[7:5] = 110 => DTF = (32+DTGF[4:0])xtdtg with Tdtg = 8xtDTS.
    • 🔖Dead Time配置为200,对应的死区时间换算:8*(32+8)/(160MHz/2)=4us
      在这里插入图片描述
  • 🌾如果上的Dead Time配置为230,其对应的二进制值:B1110 0110:
DTGF[7:5] = 111 => DTF = (32+DTGF[4:0])xtdtg with Tdtg = 16xtDTS
    • 🔖Dead Time配置为230,对应的死区时间换算:16*(32+6)/(160MHz/2)=7.6us
      在这里插入图片描述

📙STM32CubeMX串口DMA配置

  • 🔖主要时为了使用VOFA串口调试工具,来显示图形数据观测。如果自己有逻辑分析仪的话,这一点可以忽略。
  • 🌿配置串口并开启DMA:
    在这里插入图片描述
    在这里插入图片描述
  • ⚡开启DMA一定要勾选对应中断:
    在这里插入图片描述

📘业务代码完善

  • 🌿重新给定各通道计数值。
  • 🌿开启PWM输出。
	TIM1->CCR1 = 2000;TIM1->CCR2 = 5000;TIM1->CCR3 = 4000;HAL_TIM_Base_Start(&htim1);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_1);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_2);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_3);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_1);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_2);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_3);
  • 🌿串口相关的代码,根据个人需求copy相应内容:(注意勾选Use microLIB选项)
#include <stdio.h>
#include "usart.h"
#include <string.h>
#include <stdarg.h>
//使用printf()发送数据,需要对printf函数进行重定向,且只能使用USART1。
// 重定向fputc函数,使用printf()发送数据
int fputc(int ch, FILE *f)
{// 参数1:串口句柄,参数2:要发送的数据;参数3:要发生数据的长度;参数4:超时等待时间HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 100);return ch;
}
//DMA发送:本项目中使用的方法
// DMA模式
void log_DMA(const char *format, ...)
{va_list args;			// 定义参数列表变量va_start(args, format); // 从format位置开始接收参数表,放在arg里面char strBuf[256];				// 定义输出的字符串vsprintf(strBuf, format, args); // 使用vsprintf将格式化的数据写入缓冲区va_end(args);					// 结束可变参数的使用// 等待上次的数据发送完成,避免新的数据覆盖正在传输的数据,导致混乱while (HAL_UART_GetState(&huart1) == HAL_UART_STATE_BUSY_TX){// Wait for DMA transfer to complete}HAL_UART_Transmit_DMA(&huart1, (uint8_t *)strBuf, strlen(strBuf));
}//中断式发送:
// 中断模式
void log_IT(const char *format, ...)
{va_list args;			// 定义参数列表变量va_start(args, format); // 从format位置开始接收参数表,放在arg里面char strBuf[256];				// 定义输出的字符串vsprintf(strBuf, format, args); // 使用vsprintf将格式化的数据写入缓冲区va_end(args);					// 结束可变参数的使用// 等待上次的数据发送完成,避免新的数据覆盖正在传输的数据,导致混乱while (HAL_UART_GetState(&huart1) == HAL_UART_STATE_BUSY_TX){// Wait for transfer to complete}HAL_UART_Transmit_IT(&huart1, (uint8_t *)strBuf, strlen(strBuf));
}//使用sprintf()函数,阻塞式发送:
// 堵塞模式
void log(const char *format, ...)
{va_list args;			// 定义参数列表变量va_start(args, format); // 从format位置开始接收参数表,放在arg里面char strBuf[256];				// 定义输出的字符串vsprintf(strBuf, format, args); // 使用vsprintf将格式化的数据写入缓冲区va_end(args);					// 结束可变参数的使用HAL_UART_Transmit(&huart1, (uint8_t *)strBuf, strlen(strBuf), HAL_MAX_DELAY);
}
  • 🌿main函数代码
int main(void)
{/* USER CODE BEGIN 1 */float temp[3];uint8_t TempData[16];/* 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_DMA_Init();MX_TIM1_Init();MX_USART1_UART_Init();/* USER CODE BEGIN 2 */
//	TIM1->PSC = 30000;//VOFA上位机观测,开启
//	TIM1->ARR = 10000;TIM1->CCR1 = 2000;TIM1->CCR2 = 5000;TIM1->CCR3 = 4000;HAL_TIM_Base_Start(&htim1);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_1);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_2);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_3);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_1);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_2);HAL_TIMEx_PWMN_Start(&htim1,TIM_CHANNEL_3);/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1){/* USER CODE END WHILE *//* USER CODE BEGIN 3 */if((GPIOC->IDR & GPIO_PIN_0) != 0){temp[0]=1.0f;}else{temp[0]=0.0f;}if((GPIOC->IDR & GPIO_PIN_1) != 0){temp[1]=3.0f;}else{temp[1]=2.0f;}if((GPIOC->IDR & GPIO_PIN_2) != 0){temp[2]=5.0f;}else{temp[2]=4.0f;}TempData[12] = 0x00;//写入结尾数据TempData[13] = 0x00;TempData[14] = 0x80;TempData[15] = 0x7f;memcpy(TempData,(uint8_t*)temp,sizeof(temp));
//		HAL_UART_Transmit(&huart1, (uint8_t *)TempData, 16, 100);HAL_UART_Transmit_DMA(&huart1, (uint8_t *)TempData, 16);}/* USER CODE END 3 */
}

今天写不完了,明天接着写。。。

这篇关于HAL STM32G4 +TIM1 3路PWM互补输出+VOFA波形演示的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/837873

相关文章

顺序表之创建,判满,插入,输出

文章目录 🍊自我介绍🍊创建一个空的顺序表,为结构体在堆区分配空间🍊插入数据🍊输出数据🍊判断顺序表是否满了,满了返回值1,否则返回0🍊main函数 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞+关注+评论+收藏(一键四连)哦~ 🍊自我介绍   Hello,大家好,我是小珑也要变强(也是小珑),我是易编程·终身成长社群的一名“创始团队·嘉宾”

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出 在数字化时代,文本到语音(Text-to-Speech, TTS)技术已成为人机交互的关键桥梁,无论是为视障人士提供辅助阅读,还是为智能助手注入声音的灵魂,TTS 技术都扮演着至关重要的角色。从最初的拼接式方法到参数化技术,再到现今的深度学习解决方案,TTS 技术经历了一段长足的进步。这篇文章将带您穿越时

研究人员在RSA大会上演示利用恶意JPEG图片入侵企业内网

安全研究人员Marcus Murray在正在旧金山举行的RSA大会上公布了一种利用恶意JPEG图片入侵企业网络内部Windows服务器的新方法。  攻击流程及漏洞分析 最近,安全专家兼渗透测试员Marcus Murray发现了一种利用恶意JPEG图片来攻击Windows服务器的新方法,利用该方法还可以在目标网络中进行特权提升。几天前,在旧金山举行的RSA大会上,该Marcus现场展示了攻击流程,

如何将一个文件里不包含某个字符的行输出到另一个文件?

第一种: grep -v 'string' filename > newfilenamegrep -v 'string' filename >> newfilename 第二种: sed -n '/string/!'p filename > newfilenamesed -n '/string/!'p filename >> newfilename

Detectorn2预训练模型复现:数据准备、训练命令、日志分析与输出目录

Detectorn2预训练模型复现:数据准备、训练命令、日志分析与输出目录 在深度学习项目中,目标检测是一项重要的任务。本文将详细介绍如何使用Detectron2进行目标检测模型的复现训练,涵盖训练数据准备、训练命令、训练日志分析、训练指标以及训练输出目录的各个文件及其作用。特别地,我们将演示在训练过程中出现中断后,如何使用 resume 功能继续训练,并将我们复现的模型与Model Zoo中的

第六章习题11.输出以下图形

🌏个人博客:尹蓝锐的博客 希望文章能够给到初学的你一些启发~ 如果觉得文章对你有帮助的话,点赞 + 关注+ 收藏支持一下笔者吧~ 1、题目要求: 输出以下图形

Collection的所有的方法演示

import java.util.ArrayList;import java.util.Collection;import java.util.Iterator;public class TestCollection {/*** @param args* Collection的所有的方法演示* 此程序没有使用泛型,所以可以添加任意类型* 以后如果写到泛型会补充这一方面的内容*/public s

LibSVM学习(五)——分界线的输出

对于学习SVM人来说,要判断SVM效果,以图形的方式输出的分解线是最直观的。LibSVM自带了一个可视化的程序svm-toy,用来输出类之间的分界线。他是先把样本文件载入,然后进行训练,通过对每个像素点的坐标进行判断,看属于哪一类,就附上那类的颜色,从而使类与类之间形成分割线。我们这一节不讨论svm-toy怎么使用,因为这个是“傻瓜”式的,没什么好讨论的。这一节我们主要探讨怎么结合训练结果文件

下载/保存/读取 文件,并转成流输出

最近对文件的操作又熟悉了下;现在记载下来:学习在于 坚持!!!不以细小而不为。 实现的是:文件的下载、文件的保存到SD卡、文件的读取输出String 类型、最后是文件转换成流输出;一整套够用了; 重点: 1:   操作网络要记得开线程; 2:更新网络获取的数据 切记用Handler机制; 3:注意代码的可读性(这里面只是保存到SD卡,在项目中切记要对SD卡的有无做判断,然后再获取路径!)

类和对象的定义和调用演示(C++)

我习惯把类的定义放在头文件中 Student.h #define _CRT_SECURE_NO_WARNINGS#include <string>using namespace std;class student{public:char m_name[25];int m_age;int m_score;char* get_name(){return m_name;}int set_name