博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
增量式 PID 控制demo
阅读量:4166 次
发布时间:2019-05-26

本文共 6068 字,大约阅读时间需要 20 分钟。

 记录一下,以后用得着

#include "sys.h"#include "BLDC.h" #include "stm32f10x.h"#define START_PWM_PUSSE		100#define PWM_MAX_ARR	999#define PWM_MIN_ARR	49void MIX_PIN_Init(void){     GPIO_InitTypeDef  GPIO_InitStructure; 	    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);	 //使能PB,PE端口时钟	    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_8;				 //LED0-->PB.5 端口配置    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 //推挽输出    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		 //IO口速度为50MHz    GPIO_Init(GPIOC, &GPIO_InitStructure);					 //根据设定参数初始化GPIOB.5	    GPIO_ResetBits(GPIOC,GPIO_Pin_6|GPIO_Pin_8);						 //PB.5 输出高} /*20~30Khz    999 2 72000000/(1000*3)=24000*/void TIM8_PWM_Init(void){ 	GPIO_InitTypeDef GPIO_InitStructure;	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;	TIM_OCInitTypeDef  TIM_OCInitStructure;		RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM8, ENABLE);	//使能定时器3时钟 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC  | RCC_APB2Periph_AFIO, ENABLE);  //使能GPIO外设和AFIO复用功能模块时钟       //设置该引脚为复用输出功能,输出TIM3 CH2的PWM脉冲波形	GPIOB.5	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; //TIM_CH2	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	GPIO_Init(GPIOC, &GPIO_InitStructure);//初始化GPIO    //初始化TIM3	TIM_TimeBaseStructure.TIM_Period = 999; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	TIM_TimeBaseStructure.TIM_Prescaler =2; //设置用来作为TIMx时钟频率除数的预分频值 	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式	TIM_TimeBaseInit(TIM8, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位		//初始化TIM3 Channel2 PWM模式	 	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; //选择定时器模式:TIM脉冲宽度调制模式2 	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高	TIM_OCInitStructure.TIM_Pulse = 100; 	TIM_OC2Init(TIM8, &TIM_OCInitStructure);  //根据T指定的参数初始化外设TIM3 OC2	TIM_OC2PreloadConfig(TIM8, TIM_OCPreload_Enable);  //使能TIM3在CCR2上的预装载寄存器 	TIM_Cmd(TIM8, ENABLE);  //使能TIM3}/*100ms*/void TIM7_BASE_Init(void){  	NVIC_InitTypeDef        NVIC_InitStructure;	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM7, ENABLE);		TIM_TimeBaseStructure.TIM_Period = 999; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	 80K	TIM_TimeBaseStructure.TIM_Prescaler =7199; //设置用来作为TIMx时钟频率除数的预分频值  不分频	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式	TIM_TimeBaseInit(TIM7, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位		NVIC_InitStructure.NVIC_IRQChannel = TIM7_IRQn;  //TIM3中断	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;  //先占优先级0级	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;  //从优先级3级	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能	NVIC_Init(&NVIC_InitStructure);  //初始化NVIC寄存器	  TIM_ARRPreloadConfig(TIM7, ENABLE); //使能TIMx在ARR上的预装载寄存器		TIM_ITConfig(TIM7,TIM_IT_Update,ENABLE );	TIM_Cmd(TIM7, DISABLE);  //使能TIMx外设}void TIM1_ERT_Init(void){  	GPIO_InitTypeDef GPIO_InitStructure;	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;	RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE); 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE  | RCC_APB2Periph_AFIO, ENABLE);  //使能GPIO外设和AFIO复用功能模块时钟使能	GPIO_PinRemapConfig(GPIO_FullRemap_TIM1, ENABLE);		GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	GPIO_Init(GPIOE, &GPIO_InitStructure);		TIM_TimeBaseStructure.TIM_Period = 9999;	TIM_TimeBaseStructure.TIM_Prescaler =0;	TIM_TimeBaseStructure.TIM_ClockDivision = 0;	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;	TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);	TIM_ETRClockMode2Config(TIM1,TIM_ExtTRGPSC_OFF,TIM_ExtTRGPolarity_NonInverted,0);	TIM_SetCounter(TIM1, 0); 	TIM_Cmd(TIM1, DISABLE);}//增量式PIDtypedef struct PID {	float SetPoint;//目标 
<速度 pps 每秒检测到的脉冲>
float Proportion;//比例系数 float Integral;//积分系数 float Derivative;//微分系数 float LastError;上上次 float PrevError;//上次 float SumError;//累计 uint16_t CCR;//当前控制量 PWM占空比 int Set_v;//设置的目标速度
<单位 pps 每秒检测到的脉冲>
uint8_t stop_sta;} PID;PID vPID={3.0,0.4,0.001,0,0,0,0,START_PWM_PUSSE,100,0};float PIDCalc( PID *pp, float NextPoint ){ float dError, Error; Error = pp->SetPoint - NextPoint;//本次误差 目标-当前 比例 pp->SumError += Error;//累计误差 积分 dError = pp->LastError - pp->PrevError;//两次误差差值 微分 pp->PrevError = pp->LastError; //更新 pp->LastError = Error; //更新 if(pp->SumError>900) pp->SumError=900;//积分控制 else if(pp->SumError<-900) pp->SumError=-900;//积分控制 //输出增量 return (pp->Proportion * Error + pp->Integral * pp->SumError + pp->Derivative * dError );}float Vol(int vSet,int v3){ float vOut; float vIn; vPID.SetPoint = vSet; vIn = (float)v3; vOut = PIDCalc ( &vPID,vIn ); return vOut;}//100msvoid TIM7_IRQHandler(){ int v;//每秒脉冲 float Pv=0.0; if(TIM_GetITStatus(TIM7,TIM_FLAG_Update)) { TIM_ClearITPendingBit(TIM7, TIM_FLAG_Update); v = TIM1->CNT *10; TIM1->CNT = 0; if(vPID.Set_v-v>=vPID.Set_v/6 || vPID.Set_v-v<=-vPID.Set_v/6)// 当前v 大于 ±设置v { Pv=Vol(vPID.Set_v,v); vPID.CCR=vPID.CCR+Pv;//增量式 if(vPID.CCR>=PWM_MAX_ARR) { vPID.CCR=PWM_MAX_ARR; } else if(vPID.CCR<=PWM_MIN_ARR) { vPID.CCR=PWM_MIN_ARR; } TIM8->CCR2 = vPID.CCR; } if(v==0)vPID.stop_sta=0; else vPID.stop_sta=1; }}void Mix_MT_Config(){ MIX_PIN_Init(); TIM8_PWM_Init(); TIM7_BASE_Init(); TIM1_ERT_Init();}//v r/s // 300 0.05~5void Mix_MT_Start(float v){ if(v<0.05)v=0.05; else if(v>5)v=5; vPID.Set_v = v*10*6; GPIO_SetBits(GPIOC,GPIO_Pin_8); TIM_CtrlPWMOutputs(TIM8,ENABLE); TIM_SetCounter(TIM1, 0); TIM_Cmd(TIM1, ENABLE); TIM_Cmd(TIM7, ENABLE); TIM_Cmd(TIM8, ENABLE);}//v r/s 转/秒//(0.05,5)void Mix_MT_SetSpeed(float v){ if(v<0.05)v=0.05; else if(v>5)v=5; vPID.Set_v = v;}void Mix_MT_Stop(){ GPIO_ResetBits(GPIOC,GPIO_Pin_8); TIM_CtrlPWMOutputs(TIM8,DISABLE); TIM_Cmd(TIM1, DISABLE); TIM_Cmd(TIM7, DISABLE); TIM_Cmd(TIM8, DISABLE);}

 

 

转载地址:http://xqoxi.baihongyu.com/

你可能感兴趣的文章
如果指针指向了文件中的内容,外部修改了此指针指向地址的内容,文件也会修改
查看>>
进程间通信之:socket
查看>>
本地socket与网络socket实现
查看>>
linux中的find命令——查找文件名
查看>>
linux系统下 CP命令(备注:替换正在运行的进程,用FTP工具是从本地传是不行的,需要用cp -rf ../update/zb_terminal ./zb_terminal)
查看>>
linux进程运行的相关shell命令 fg、bg、jobs、&、ctrl + z(后台、前台等等)
查看>>
linux ls命令ls /dev | grep sda 什么意思(以及grep连接符|怎么用)
查看>>
system()、popen()
查看>>
linux mount (挂载命令)、umount (卸载命令)详解
查看>>
chmod命令详解使用格式和方法
查看>>
linux中的access函数
查看>>
chdir改变当前目录以及理解守护进程为何fork()两次?
查看>>
创建守护进程以及为什么fork两次
查看>>
open/close函数的使用
查看>>
C语言中lseek()函数和fseek()函数的使用详解
查看>>
linux 中 open和fopen的区别
查看>>
LINUX共享内存使用常见陷阱与分析
查看>>
Linux编程gcc编译器禁止所有警告和显示所有警告
查看>>
共享内存理解:shmget()函数、shmat()、IPC_CREAT | IPC_EXCL
查看>>
ctime函数--把日期和时间转换为字符串(打印时间时常用-LINUX系统)
查看>>