FPGA实现电机位置环、速度环双闭环PID控制

一、设计思路

主要设计思路就是根据之前写的一篇FPGA实现电机转速PID控制,前面已经实现了位置环的控制,思想就是通过电机编码器的当前位置值不断地修正PID去控制速度。

那为了更好的实现控制,可以在位置环后加上速度环,实现电机位置环、速度环双闭环PID控制。

位置环作为外环,通过编码器计数通过PID输出速度;位置环输出的速度作为目标速度输入速度环,与编码器测速的当前速度进行PID计算,从而完成电机的双PID控制。

​​

二、位置环控制

位置环的控制在前面已经实现,再次不再赘述。

三、速度环控制

速度环作为内环,目标值为位置环输出的速度,当前值为编码器测速的速度。

这里就有了一个问题,编码器怎么样才能测到电机速度?

这就和电机的最大频率有关,编码器通过AB相位的波形去进行计数和正反转的判断,那么就需要知道电机转一圈所输出的脉冲个数以及空载最大转速,通过这两个参数计算出编码器的电机最大频率,即每秒会有多少个脉冲。因此要对时钟进行分频,这是对50MHZ时钟分频成100HZ,即时钟周期10ms进行一次PID输出。

cpp 复制代码
always @(posedge clk or negedge reset_n) begin
   if (~reset_n) begin
      clk100 <= 1'b0;
      clk_counter <= 32'd0;
   end else if (clk_counter == 32'd249999)begin
      //分频系数M=时钟输入频率/时钟输出频率,偶分频计数器值N=M/2;奇分频计数器值N=M-1/2
      clk100 <= ~clk100;
      clk_counter <= 32'd0;
   end else begin
      clk_counter <= clk_counter + 1;
   end
end

速度环的PID计算同位置环一样

cpp 复制代码
always @(posedge clk100 or negedge reset_n) begin
   if (~reset_n) begin
      error      <= 0;
      integral   <= 0;
      prev_error <= 0;
      derivative <= 0;
      pidoutput  <= 0;
   end else begin

      //target位置环输出的速度,current当前编码器值
      error <= target - current;

      // 将累积误差控制合理范围内
      integral <= integral + error;
      if(integral >= $signed(32'd500)) begin
         integral <= $signed(32'd500);
      end else if(integral <= $signed(-32'd500M)) begin
         integral <= $signed(-32'd500);
      end else begin
         integral <= integral;
      end

      derivative <= error - prev_error;

      // 计算PID输出
      p <= error * kp;
      i <= integral * ki;
      d <= derivative * kd;
      pidoutput <= (p + i + d) / 1000;
      
   end
end

四、编码器测速

编码器模块也需要加入测速代码,有两种测速方法:一种是两次位置值相减,得出电机速度;另一种是编码器计数,得出电机速度。

这里用第二种编码器计数,在速度环中每10ms输出一次PID值,那么在测速模块就每10ms将速度编码器置零重新计数,因此可以根据速度环传入的清零信号clear进行清零,即clk100。当clk100状态发生变化时,clear信号为1,清零速度计数。

cpp 复制代码
always@(posedge clk or negedge reset_n) begin
	if(!reset_n) begin
		delay0 <= 1'b0;
		delay1 <= 1'b0;		
	end else begin
		delay0 <= clk100;
		delay1 <= delay0;
	end
end
assign clear  = (delay0 & (~delay1));

always @(posedge clk or negedge reset_n) begin
	if (~reset_n) begin
		counter <= 0;
	end
	else if (clear) begin
		counter <= 0;
	end 
	else if(pulse) begin
		if(direction) begin
			if(counter < $signed(32'h8000))
				counter <= counter + 1;
		end 
		else if(!direction) begin
			if(counter > $signed(-32'h8000))
				counter <= counter - 1;
		end 
		else begin
			counter <= 0;     
		end   
	end
end

五、整体结构

整体的RTL图如图所示。

相关推荐
luckyluckypolar7 分钟前
STM32 -中断
stm32·单片机·嵌入式硬件·物联网
皇华ameya3 小时前
AMEYA360:村田电子更适合薄型设计应用场景的3.3V输入、12A输出的DCDC转换IC
fpga开发
启明云端wireless-tag4 小时前
ESP32无线WiFi蓝牙SOC,设备物联网通信方案,启明云端乐鑫代理商
嵌入式硬件·物联网·wifi·esp32·乐鑫·wifi模组
千穹凌帝5 小时前
SpinalHDL之结构(二)
开发语言·前端·fpga开发
@@庆5 小时前
stm32 PWR电源控制(修改主频&睡眠模式&停机模式&待机模式)
stm32·单片机·嵌入式硬件
JT灬新一5 小时前
STM32巡回研讨会总结(2024)
stm32·单片机·嵌入式硬件
爱桥代码的程序媛6 小时前
鸿蒙OpenHarmony【轻量系统芯片移植案例】标准系统方案之瑞芯微RK3568移植案例
嵌入式硬件·harmonyos·鸿蒙·鸿蒙系统·移植·openharmony·鸿蒙开发
Whappy0017 小时前
51单片机-DA(数字转模拟)
单片机·嵌入式硬件·51单片机
鸽子汤1977 小时前
想高效开发?从文件系统开始着手。。。
嵌入式硬件·物联网·硬件工程
Whappy0017 小时前
51单片机-AD(模拟信号转数字信号)-实验()
单片机·嵌入式硬件·51单片机