GD32F103RCT6/GD32F303RCT6(9)高级定时器互补PWM波输出实验

本文章基于兆易创新GD32 MCU所提供的2.2.4版本库函数开发

       后续项目主要在下面该专栏中发布:

手把手教你嵌入式国产化_不及你的温柔的博客-CSDN博客

       感兴趣的点个关注收藏一下吧!

       电机驱动开发可以跳转:

手把手教你嵌入式国产化-实战项目-无刷电机驱动(1)-CSDN博客

       BMS电源系统开发可以跳转:暂未放链接

 向上代码兼容GD32F303RCT6中使用

本项目配套开发板:

基于GD32F103RCT6国产GD32平台,以下教程编写基于该开发板

图片:

a83f44e3ba7542238ec2f3c3d9002bbe.jpeg

原理图以及例程请联系客服获取!

注意:

本教程致力于解决所有在调试中出现的所有问题,如有未包含在的问题,请联系QQ:2049363803,有奖更新文档!

定时器介绍

在上一节中,我们已经完成了对高级/通用定时器输出PWM波的实验

8fd924e596df4992b0e4fbed6aba263b.png

本次实验将会研究高级定时器TIMER0的互补PWM输出

这里需要强调一下,和串口一样,GD32的定时器定义是从0为编号开始的!

在逻辑框图中,关于互补波形输出的控制是在右下角的输出逻辑中实现的

寄存器

涉及到的寄存器主要是以下几个:

常规的时钟源配置和时钟预分频器、计数模式等在上一篇文章中有介绍就不作为主要讲解,今天主要是是关于互补寄存器。

控制寄存器 1TIMERx_CTL1

主要涉及到互补波形的空闲状态电平以及通道的触发源选择和模式控制,其次就是DMA以及影子寄存器的控制这里使用不到就不做讲解;

第八位和第九位的ISO0以及ISO0N作为主要控制位。

通道控制寄存器 2TIMERx_CHCTL2

涉及到互补的主要配置:有效极性、使能状态等,该寄存器是读写寄存器,意味着我们可以随时读取以获取通道使能状态或者写入数据使得通道启用或者禁用、包括在运行过程中改变输出互补波形的极性状态。

互补PWM实验

编程要点:

1.初始化结构体

2.使能GPIO时钟

3.设置GPIO工作模式

4.使能复用时钟和定时器时钟

5.初始化定时器参数

6.配置互补通道参数

7.使能通道输出

bsp_timer.c

#include "bsp_timer.h"
#include "bsp_led.h"
#include "oled.h"



//GPIO管脚配置
void gpio_config(void)
{	  
		rcu_periph_clock_enable(RCU_GPIOA);
    rcu_periph_clock_enable(RCU_GPIOB);
    rcu_periph_clock_enable(RCU_AF);

    /*Configure PA8 PA9 PA10(TIMER0 CH0 CH1 CH2) as alternate function*/
    gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8);
    gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_9);
    gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_10);

    /*Configure PB13 PB14 PB15(TIMER0 CH0N CH1N CH2N) as alternate function*/
    gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_13);
    gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_14);
    gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_15);
			
}

/**
 @brief PWM初始化
 @param 无
 @return 无
*/

//定时器配置,TIM1,通道1,通道2,通道3
void timer_config(uint16_t arr,uint16_t psc)
{
	gpio_config();
	

	
	  rcu_periph_clock_enable(RCU_TIMER0);
	
    timer_oc_parameter_struct timer_ocintpara;
	  timer_oc_parameter_struct timer_ocintpara1;
	  timer_oc_parameter_struct timer_ocintpara2;
	
    timer_parameter_struct timer_initpara;

    

    timer_deinit(TIMER0);

    /* TIMER0 configuration */
    timer_initpara.prescaler         = psc;
    timer_initpara.alignedmode       = TIMER_COUNTER_EDGE;
    timer_initpara.counterdirection  = TIMER_COUNTER_UP;
    timer_initpara.period            = arr;
    timer_initpara.clockdivision     = TIMER_CKDIV_DIV1;
    timer_initpara.repetitioncounter = 0;
    timer_init(TIMER0,&timer_initpara);

     /* CH1,CH2 and CH3 configuration in PWM mode */
    timer_ocintpara.outputstate  = TIMER_CCX_ENABLE;
    timer_ocintpara.outputnstate = TIMER_CCXN_ENABLE;
    timer_ocintpara.ocpolarity   = TIMER_OC_POLARITY_HIGH;
    timer_ocintpara.ocnpolarity  = TIMER_OCN_POLARITY_HIGH;
    timer_ocintpara.ocidlestate  = TIMER_OC_IDLE_STATE_LOW;
    timer_ocintpara.ocnidlestate = TIMER_OCN_IDLE_STATE_LOW;
		
		timer_ocintpara1.outputstate  = TIMER_CCX_ENABLE;
    timer_ocintpara1.outputnstate = TIMER_CCXN_ENABLE;
    timer_ocintpara1.ocpolarity   = TIMER_OC_POLARITY_HIGH;
    timer_ocintpara1.ocnpolarity  = TIMER_OCN_POLARITY_HIGH;
    timer_ocintpara1.ocidlestate  = TIMER_OC_IDLE_STATE_LOW;
    timer_ocintpara1.ocnidlestate = TIMER_OCN_IDLE_STATE_LOW;
		
		timer_ocintpara2.outputstate  = TIMER_CCX_ENABLE;
    timer_ocintpara2.outputnstate = TIMER_CCXN_ENABLE;
    timer_ocintpara2.ocpolarity   = TIMER_OC_POLARITY_HIGH;
    timer_ocintpara2.ocnpolarity  = TIMER_OCN_POLARITY_HIGH;
    timer_ocintpara2.ocidlestate  = TIMER_OC_IDLE_STATE_LOW;
    timer_ocintpara2.ocnidlestate = TIMER_OCN_IDLE_STATE_LOW;
		

    timer_channel_output_config(TIMER0,TIMER_CH_0,&timer_ocintpara);
    timer_channel_output_config(TIMER0,TIMER_CH_1,&timer_ocintpara1);
    timer_channel_output_config(TIMER0,TIMER_CH_2,&timer_ocintpara2);

    timer_channel_output_pulse_value_config(TIMER0,TIMER_CH_0,500);
    timer_channel_output_mode_config(TIMER0,TIMER_CH_0,TIMER_OC_MODE_PWM0);
    timer_channel_output_shadow_config(TIMER0,TIMER_CH_0,TIMER_OC_SHADOW_DISABLE);

    timer_channel_output_pulse_value_config(TIMER0,TIMER_CH_1,500);
    timer_channel_output_mode_config(TIMER0,TIMER_CH_1,TIMER_OC_MODE_PWM0);
    timer_channel_output_shadow_config(TIMER0,TIMER_CH_1,TIMER_OC_SHADOW_DISABLE);

    timer_channel_output_pulse_value_config(TIMER0,TIMER_CH_2,500);
    timer_channel_output_mode_config(TIMER0,TIMER_CH_2,TIMER_OC_MODE_PWM0);
    timer_channel_output_shadow_config(TIMER0,TIMER_CH_2,TIMER_OC_SHADOW_DISABLE);

    timer_primary_output_config(TIMER0,ENABLE);
    
    /* auto-reload preload enable */
    timer_auto_reload_shadow_enable(TIMER0);
    timer_enable(TIMER0);
}


//设置TIMER0通道0的占空比
//compare:比较值
void TIM_SetTIM0Compare1(uint32_t compare)
{
  timer_channel_output_pulse_value_config(TIMER0, TIMER_CH_0, compare);
}

//设置TIMER0通道0的占空比
//compare:比较值
void TIM_SetTIM0Compare2(uint32_t compare)
{
  timer_channel_output_pulse_value_config(TIMER0, TIMER_CH_1, compare);
}

//设置TIMER0通道0的占空比
//compare:比较值
void TIM_SetTIM0Compare3(uint32_t compare)
{
  timer_channel_output_pulse_value_config(TIMER0, TIMER_CH_2, compare);
}


// TIMER0更新中断服务程序
void TIMER0_UP_IRQHandler(void) {
    if (SET == timer_interrupt_flag_get(TIMER0, TIMER_INT_FLAG_UP)) 
		{
        timer_interrupt_flag_clear(TIMER0, TIMER_INT_FLAG_UP);
//			  LED1_ON; // 点亮LED
//			  LED2_ON; // 点亮LED
        // 在这里添加触发中间对齐中断的处理代码
    }
}

我们在这里设置初始化默认占空比大小

主函数中完成初始化设置

实验结果

其中黄色通道1是PA8输出的波形,通道2是PB13输出的波形,完成实验。

拓展部分

在使用GD32驱动EG2133的过程中,需要互补的波形有效电平极性能根据我的使用情况发生改变,使得互补的PWM波与主通道相同或者反相

前文我们讲到,改变互补PWM极性是更改TIMER_CHCTL2寄存器中的CH0NP位的数据

那么我们只需要根据寄存器地址寻址,对3号位置寄存器的数据进行更改就行。

上代码:
#define TIMER0_BASE_ADDRESS 0x40012C00 // TIMER0的基地址
#define TIMER_CHCTL2_OFFSET 0x20       // TIMER_CHCTL2寄存器相对于基地址的偏移
#define TIMER0_CHCTL2       (*((volatile uint32_t*)(0x40012C00 + 0x20)))

#define TIMER_CHCTL2_CH0EN_MASK (1U << 0) // CH0EN位在第0位
#define TIMER_CHCTL2_CH1EN_MASK (1U << 4) // CH1EN位在第4位
#define TIMER_CHCTL2_CH2EN_MASK (1U << 8) // CH2EN位在第8位

#define TIMER_CHCTL2_CH0NP_MASK (1U << 3) // CH0EN位在第0位
#define TIMER_CHCTL2_CH1NP_MASK (1U << 7) // CH1EN位在第4位
#define TIMER_CHCTL2_CH2NP_MASK (1U << 11) // CH2EN位在第8位

#define TIMER_CHCTL2_CH0NP_PC_0   TIMER0_CHCTL2 &= (~(1U<<3))
#define TIMER_CHCTL2_CH0NP_PC_1   TIMER0_CHCTL2 |= 1U<<3            //或只需要改变要操作的位数

#define TIMER_CHCTL2_CH1NP_PC_0   TIMER0_CHCTL2 &= (~(1U<<7))
#define TIMER_CHCTL2_CH1NP_PC_1   TIMER0_CHCTL2 |= 1U<<7       

#define TIMER_CHCTL2_CH2NP_PC_0   TIMER0_CHCTL2 &= (~(1U<<11))
#define TIMER_CHCTL2_CH2NP_PC_1   TIMER0_CHCTL2 |= 1U<<11 

想找到一个寄存器的位置,首先需要知道定时器0的总地址,然后其中每个地址对于该地址的偏移就能完成偏移寻址。

主函数代码

在mian中调用:TIMER_CHCTL2_CH0NP_PC_1;    使得由反相变成同向

示波器结果:

再次调用反相函数使得其更改为反相

完成本次实验!!

创作不易,点个赞收藏一下吧,求求了。比个爱心

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/603765.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

第 8 章 电机测速(自学二刷笔记)

重要参考&#xff1a; 课程链接:https://www.bilibili.com/video/BV1Ci4y1L7ZZ 讲义链接:Introduction Autolabor-ROS机器人入门课程《ROS理论与实践》零基础教程 8.3.3 电机测速01_理论 测速实现是调速实现的前提&#xff0c;本节主要介绍AB相增量式编码器测速原理。 1.概…

深度学习——前馈全连接神经网络

前馈全连接神经网络 1.导入需要的工具包2.数据导入与数据观察&#xff08;1&#xff09;读取csv的文件信息&#xff1a;&#xff08;2&#xff09;训练数据前5行&#xff08;3&#xff09;打印第一个图&#xff08;4&#xff09;观察数据中的信息&#xff08;5&#xff09;查看…

浏览器输入URL到页面展示的过程详解

重点面试题&#xff1a;当你的浏览器中地址栏输入地址并回车的一瞬间到页面能够展示回来&#xff0c;经历了什么&#xff1f; step 1、URL解析 URL&#xff1a;internet上的每一个网页都具有一个唯一的名称标识&#xff0c;通常称之为URL&#xff08;Uniform Resource Locator…

2024蓝桥杯网络安全部分赛题wp

爬虫协议 题目给了提示访问robots.txt 会出三个目录 访问最后一个 点进去就flag{22560c15-577c-4c8b-9944-815473758bad} packet 下载附件&#xff0c;这个是流量包 放wireshark流量分析 搜http协议 发现有cat flag命令&#xff0c;直接看他返回的流量 最后base64解码即可…

C++|多态(虚函数、抽象类、多态原理)

目录 一、多态的概念及构成 1.1概念 1.2多态的构成条件(继承虚函数) 二、虚函数和虚函数重写 2.1虚函数和虚函数重写的概念 2.2虚函数的"异变"(协变析构重写) 2.3虚函数的扩展(overridefinal) 2.4重载、重写(覆盖)、隐藏(重定义)的对比 三、抽象类 3.1概念…

43.乐理基础-拍号-常见的拍号与强弱关系

首先拍号的定义&#xff1a;39.认识音符、40.什么是一拍、41.小节、小节线、终止线、42.看懂拍号的意义 通过 39.认识音符、40.什么是一拍、41.小节、小节线、终止线、42.看懂拍号的意义 应该可以知道 Y的取值只能是2、4、8、16、32、64。。。。因为Y指的是Y分音符&#xff0c;…

树莓派4b测量PM2.5

1.GP2Y1010AU0F粉尘传感器连接图 2. GP2Y1010AU0F工作原理 工作原理 传感器中心有个洞可以让空气自由流过,定向发射LED光,通过检测经过空气中灰尘折射过后的光线来判断灰尘的含量。 3.源代码 main.py # coding=UTF-8 import RPi.GPIO as GPIO from ADC import ADS1015…

【进程等待】是什么 | 为什么 | 怎么办 | wait阻塞等待

目录 进程等待是什么&#xff1f; 为什么要进程等待&#xff1f; 如何进程等待&#xff1f; wait 阻塞等待 进程等待是什么&#xff1f; 进程终止会把进程退出的数据&#xff08;退出码和退出信号&#xff09;存放到进程的PCB中保存下来&#xff0c;让父进程进行等待。…

OpenHarmony 实战开发 - 如何在源码中编译复杂应用(4.0-Release)

文档环境 开发环境&#xff1a;Windows 11 编译环境&#xff1a;Ubuntu 22.04 开发板型号&#xff1a;DAYU 200&#xff08;RK3568&#xff09; 系统版本&#xff1a;OpenHarmony-4.0-Release 功能简介 在 OpenHarmony 系统中预安装应用的 hap 包会随系统编译打包到镜像中&a…

Java毕业设计 基于SpringBoot vue社区智慧养老监护管理平台

Java毕业设计 基于SpringBoot vue社区智慧养老监护管理平台 SpringBoot 社区智慧养老监护管理平台 功能介绍 登录注册 个人中心 修改密码 个人信息 房间信息管理 房间入住信息管理 反馈信息管理 留言管理 老人信息管理 公告管理 物资申请管理 管理员管理 护工管理 体检员管理…

浅谈Windows 上的线程亲和性(Thread affinity)

​ 前言 线程属性包括是否分离、亲和性、调度策略和优先级等。Linux默认的调度策略是CFS(完全公平调度算法),而 Windows 是基于优先级抢占式的策略。 在这些方面,Windows 和 Linux 差异巨大。本文仅针对 Windows 系统的线程亲和性进行探讨。 线程亲和性(Thread affinity) 什…

解锁AI的神秘力量:LangChain4j带你步入智能化实践之门

关注微信公众号 “程序员小胖” 每日技术干货&#xff0c;第一时间送达&#xff01; 引言 在数字化转型的浪潮中&#xff0c;人工智能&#xff08;AI&#xff09;正逐渐成为推动企业创新和增长的关键力量。然而&#xff0c;将AI技术融入到日常业务流程并非易事&#xff0c;它…

谷歌月球模型

收费产品&#xff0c;白嫖党勿扰 收费金额500元 1 概述 前些时间&#xff0c;有个客户&#xff0c;想fight TAIWAN&#xff0c;于是乎&#xff0c;我把谷歌地球整个台湾的模型都下载下来了&#xff0c;大约300GB。今天&#xff0c;又有个客户&#xff0c;提出一个过分要求&…

【第14章】spring-mvc之ajax

文章目录 前言一、准备二、单个值1.前端2.后端3. 结果 三、对象1.前端2.后端3. 结果 四、JSON对象1.前端2.后端3. 结果 五、JSON数组1.前端2.后端3. 结果 总结 前言 AJAX&#xff08;Asynchronous JavaScript and XML&#xff09;是一种用于创建快速动态网页的技术&#xff0c…

STM32:GPIO输入输出

文章目录 1、GPIO介绍1.1 GPIO的基本结构1.1 GPIO的位结构 2、 GPIO工作模式3、GPIO标准外设库接口函数3.1 RCC接口函数3.2 GPIO接口函数3.2.1 GPIO的读取函数3.2.1 GPIO的写入函数 4、GPIO的初始化 1、GPIO介绍 GPIO&#xff08;General Purpose Input Output&#xff09;通用…

深入大模型量化技术,大模型端侧落地已Ready?

揭秘未来&#xff1a;大模型量化技术如何革新移动AI应用 ©作者|饮水机 来源|神州问学 前言 最近&#xff0c;苹果发布了OpenELM系列模型&#xff0c;参数规模分别为270M、450M、1.1B和3B。与此同时&#xff0c;微软也推出了Phi-3系列模型&#xff0c;其中mini版本的参数…

支付时,中国网联结算与中国银联结算的区别与联系

随着电子商务和互联网支付的快速发展&#xff0c;中国的支付清算市场也呈现出前所未有的繁荣景象。在这个大背景下&#xff0c;中国网联与中国银联作为两大支付清算机构&#xff0c;各自扮演着重要的角色。本文将对两者的区别和联系进行深入探讨&#xff0c;以期对读者有更全面…

【北京迅为】《iTOP-3588开发板快速烧写手册》-第9章ubuntu系统下升级固件

RK3588是一款低功耗、高性能的处理器&#xff0c;适用于基于arm的PC和Edge计算设备、个人移动互联网设备等数字多媒体应用&#xff0c;RK3588支持8K视频编解码&#xff0c;内置GPU可以完全兼容OpenGLES 1.1、2.0和3.2。RK3588引入了新一代完全基于硬件的最大4800万像素ISP&…

智慧公厕,城市现代化公卫变革的关键节点

随着城市的快速发展&#xff0c;公共厕所作为社会基础民生设施&#xff0c;正越来越受到精细化管理的重视。智慧公厕的出现&#xff0c;为传统公共厕所的脏乱臭提供了解决方案&#xff0c;通过物联网、大数据、云计算、自动化控制等先进技术的贡献&#xff0c;使公厕管理更加高…

怎么编辑百度百科个人词条

辑百度百科个人词条是一个相对复杂的过程&#xff0c;需要遵循一定的步骤和规则。以下是百科优化网整理的编辑百度百科个人词条的步骤和注意事项。 1. 确定编辑资格 百度百科个人词条的编辑权主要赋予那些具有一定影响力的公众人物&#xff0c;或者是有一定“身份”的人物&…