1
0
mirror of https://github.com/WangXuan95/FpOC.git synced 2025-01-17 23:22:52 +08:00
WangXuan95 de26548026 update
2022-04-06 20:57:18 +08:00
2022-04-05 10:34:50 +08:00
2022-04-06 20:57:18 +08:00
2022-04-01 05:23:15 +08:00
2022-04-01 05:22:10 +08:00
2022-04-05 10:34:50 +08:00

语言 仿真 部署 部署

FpOC

基于 FPGA磁场定向控制 (FOC),用于驱动永磁同步电机 (PMSM)无刷直流电机 (BLDC)

FOC控制算法对传感器采样速率和处理器算力提出了一定的要求,使用 FPGA 实现的 FOC 可以获得更好的实时性,并且更方便进行多路扩展多路反馈协同

本库实现了基于角度传感器(也就是磁编码器)的有感 FOC,即一个完整的电流环,可以进行扭矩控制。借助本库,你可以进一步使用 纯FPGAMCU+FPGA 的方式实现更复杂的电机应用。

diagram
图1:系统框图

本库代码有详细的注释,如果你熟悉 Verilog 但不熟悉 FOC ,可以通过阅读代码来快速地学习 FOC (建议先阅读 FOC 原理 [6~9])。一些用户在读代码时向本人反馈了一些疑问,我将它们整理在了 FAQ 里。

技术特点

  • 平台无关 :纯 RTL 编写,可以在 Altera 和 Xilinx 等各种 FPGA 上运行。
  • 支持 3路PWM + 1路EN PWM=1 时上桥臂 MOS 导通PWM=0 时下桥臂 MOS 导通。 EN=0 时所有的 6 个 MOS 关断。
  • 支持 12bit 分辨率角度传感器相电流采样ADC。对于>12bit的传感器需要进行低位截断。对于<12bit的传感器需要进行低位填充。
  • 内部使用 16bit 有符号整数进行计算,考虑到传感器为 12bit所以 16bit 计算是够用的。

目录

示例程序:让电机转起来

图1 是示例程序的系统框图,它调用了我实现的 FOC 电流环模块 (foc_top.sv) 来实现一个简单的行为——控制电机的电流(扭矩)按顺时针、逆时针、顺时针、逆时针地交替运行。同时,使用 UART 打印电流的控制目标值实际值,以便观察 FOC 电流环控制的质量。

该示例的所有代码都在 RTL 目录内。工程在 FPGA 目录内(这是一个 Quartus 工程,但由于本库是可移植性设计,也可以很容易地部署到 VIvado 上,只需要把不可移植的 PLL 核替换掉)。

搭建硬件

运行本库的电机驱动需要的硬件包括:

  • FPGA 开发板 :需有至少 10GPIO ,用来接:
    • I2C (2*GPIO) 用来连接 AS5600 磁编码器。
    • SPI (4*GPIO) 用来连接 AD7928 ADC。
    • PWM (3*GPIO) 用来输出 3 相 PWM 到电机驱动板。
    • PWM_EN (1*GPIO) 用来输出 1 路 EN (使能) 到电机驱动板EN=低电平代表所有桥臂关断)。
    • UART (1*GPIO) ,单向(仅发送)的UART连接计算机的串口用于监测电流环的跟随曲线可不接
  • PMSMBLDC 电机
  • 电机驱动板要支持3相 PWM 输入信号PWM=0时下桥臂导通PWM=1时上桥臂导通并且要内置低侧电阻采样+放大器对 3 相电流进行放大。
  • 获取转子角度的磁编码器:本库直接支持的型号是 AS5600 ,需要安装在电机上。
  • 相电流采样的ADC:本库直接支持的型号是 AD7928 ,用于采样电机驱动板放大后的 3 相电流
    • 网上似乎找不到现成卖的 AD7928 模块所以如果你想DIY就需要自己画 PCB。

如果你硬件动手能力很强,可以自己准备以上硬件。

不过!建议直接使用我画的一个自带 AD7928电机驱动板,它的原理图如图2,我也提供了制造文件 gerber_pcb_foc_shield.zip 你需要拿制造文件去打样PCB然后按照图2来焊接。

注意,这个板子包括了相电流采样的ADC电机驱动板的功能,但不包括 AS5600 磁编码器,磁编码器是需要安装在电机上的,需要你额外准备。

wave
图2:电机驱动板原理图,其中 AD7928 ADC 用来采样三相电流

这个板子在立创EDA开源oshwhub.com/wangxuan/arduino-foc-shield

这个板子之所以被设计成 Arduino 扩展板的样子,是因为很多 FPGA 开发板也具有 Arduino 扩展板接口(比如 DE10-Nano 开发板),可以直接插上去。如果你的 FPGA 开发板没有 Arduino 扩展板接口也没关系,直接用杜邦线连接即可(建议用短的杜邦线)。

这里使用的电机驱动器是 MP6540 芯片, 我只试过驱动小功率云台电机,没试过大功率电机。

建立FPGA工程

你需要建立 FPGA 工程,把 RTL 目录(包括其子目录)里的所有 .sv 源文件加入工程,请以 fpga_top.sv 作为顶层文件。

时钟配置

fpga_top.sv 中有一处调用了 altpll 原语,用来把开发板晶振输入的 50MHz 时钟clk_50m 信号)变成 36.864MHz 的主时钟clk 信号altpll 原语只适用于 Altera Cyclone IV FPGA如果你用的是其它系列的 FPGA需要使用它们各自的 IP 核或原语(例如 Xilinx 的 clock wizard替换它。

若你的开发板的晶振不是 50MHz ,你需要修改 PLL 的配置,保证主时钟 clk 信号的频率是 36.864MHz 即可。

实际上,主时钟 clk 的频率可以取小于 40MHz 的任意值。clk 是 FOC 系统的驱动时钟clk 的频率会决定 SVPWM 的频率SVPWM频率=clk 频率/2048我选 36.864MHz 是因为可以让 SVPWM 频率 = 36864/2048 = 18kHz只是为了凑个整数。

clk 的频率不能超过 40MHz 的原因是 adc_ad7928.sv 会通过二分频来产生 SPI 时钟spi_sck而 ADC7928 芯片要求 SPI 时钟不能超过 20MHz。

引脚约束

fpga_top.sv 的 IO 连接方法如下:

  • clk_50m 连接在 FPGA 开发板的晶振上。
  • i2c_scl, i2c_sda 连接 AS5600 (磁编码器) 的 I2C 接口。
  • spi_ss, spi_sck, spi_mosi, spi_miso 连接 AD7928 (ADC芯片) 的 SPI 接口。
  • pwm_a, pwm_b, pwm_c 连接电机驱动板的 3 相 PWM 信号。
  • pwm_en : 连接电机驱动板的 EN (使能) 信号。
    • 如果电机驱动板没有 EN 输入,则不接。
    • 如果电机驱动板有 3 路 EN 输入,每路对应 1 相,则应该进行一对三连接。
  • uart_tx 连接 UART 转 USB 模块,插入计算机的 USB 口,用于监测电流环的跟随曲线,可不接

调参

要让电机正常工作,你需要在 fpga_top.sv 中的第103行开始根据实际情况调整 foc_top 模块的参数Verilog parameter包括

  • INIT_CYCLES 决定了初始化步骤占多少个时钟(clk)周期取值范围为1~4294967294。该值不能太短因为要留足够的时间让转子回归电角度=0。在clk频率为 36.864MHz 的情况下,我们可以取 INIT_CYCLES=16777216则初始化时间为 16777216/36864000=0.45 秒。
  • ANGLE_INV
    • 若角度传感器没装反A相→B相→C相→A相 的旋转方向与角度传感器的读值的增大方向相同),则该参数应取 0
    • 若角度传感器装反了A相→B相→C相→A相 的旋转方向与 角度传感器的读值的增大方向相反),则该参数应取 1
  • POLE_PAIR电机极对数取值范围1~255根据电机型号决定注意电角度ψ = 极对数N * 机械角度φ)
  • MAX_AMPSVPWM 的最大振幅取值范围为1~511该值越小电机能达到的最大力矩越小但考虑到使用3相下桥臂电阻采样法来采样电流该值也不能太大以保证3个下桥臂有足够的持续导通时间来供ADC进行采样。在本例中使用默认值 9'd384 即可。
  • SAMPLE_DELAY采样延时取值范围0~511考虑到3相的驱动 MOS 管从开始导通到电流稳定需要一定的时间所以从3个下桥臂都导通到 ADC 采样时刻之间需要一定的延时。该参数决定了该延时是多少个时钟周期,当延时结束时,该模块在 sn_adc 信号上产生一个高电平脉冲,指示外部 ADC “可以采样了”。在本例中,使用默认值 9'd120 即可。
  • KpPID 的 P 参数。
  • KiPID 的 I 参数。

调好参后,综合并烧录到 FPGA 后,应该能看到电机正反交替运行。

用串口监视电流环

把 uart_tx 信号通过 UART 转 USB 模块 (例如 CP2102、CH340 模块) 连接到电脑上,就可以用串口助手Putty等软件来监测电流环的跟随效果。

注: UART 的格式是 115200,8,n,1

以下是串口打印的部分信息。其中第1~4列分别为d轴电流的实际值d轴电流的目标值q轴电流的实际值q轴电流的目标值。可以看到,即使目标值从+200突变到-200实际值能跟着目标值走说明电流环的 PID 控制是有效的。

 -5       0     206     200 
-16       0     202     200 
 16       0     192     200 
 15       0     201     200 
  1       0     197     200 
 17       0    -211    -200 
 -6       0    -199    -200 
-10       0    -210    -200 
 -3       0    -207    -200 
  0       0    -202    -200 
-15       0    -211    -200 

另外,你可以借用 Arduino IDE串口绘图器来实时显示电流跟随曲线。前往Arduino官网下载 Arduino IDE,安装后打开,在“工具→端口”中选择正确的COM口然后点击“工具→串口绘图器”,串口绘图器会自动接收串口并使用上述4列数据画实时曲线图。

图3 是我这里绘制出的电流跟随曲线。蓝色曲线是第1列数据d轴电流的实际值红色曲线是第2列数据d轴电流的目标值绿色曲线是第3列数据q轴电流的实际值土黄色曲线是第4列数据q轴电流的目标值。可以看到实际值能跟着目标值走。

wave
图3:电流跟随曲线

设计代码详解

下表罗列了该工程使用的所有 SystemVerilog 代码文件,这些文件都在 RTL 目录下。结合图1就能看出每个模块的作用。

文件名 功能 备注
fpga_top.sv FPGA工程的顶层模块
uart_monitor.sv UART 发送器,用于数据监测 不需要的话可以移除
i2c_register_read.sv I2C 读取器,用于读取 AS5600 磁编码器
adc_ad7928.sv AD7928 读取器
foc_top.sv FOC+SVPWM (即图1中的蓝色部分的顶层) 固定功能,一般不需要改动
clark_tr.sv Clark 变换 固定功能,一般不需要改动
park_tr.sv Park 变换 固定功能,一般不需要改动
sincos.sv 正弦/余弦计算器,被 park_tr.sv 调用 固定功能,一般不需要改动
pi_controller.sv PI 控制器PID没有D 固定功能,一般不需要改动
cartesian2polar.sv 直角坐标系转极坐标系 固定功能,一般不需要改动
svpwm.sv SVPWM 调制器 固定功能,一般不需要改动
hold_detect.sv 监测3个下桥臂都导通时延迟一段时间后触发 sn_adc 信号指示ADC可以开始采样 固定功能,一般不需要改动
diagram
图1:系统框图

图1展示了这些模块的层次,我在设计模块层次时充分考虑了封装的合理性和代码重用

  • 粉色部分是FPGA内的即传感器控制器硬件相关逻辑,如果角度传感器和 ADC 型号变了,这部分代码需要重写。
  • 蓝色部分是FPGA内的 FOC 的固定算法,是硬件无关逻辑,一般不需要修改,是本库的核心代码!
  • 黄色部分是FPGA内的用户自定逻辑,用户可以修改 user behavior 来实现各种电机应用。或者修改 uart_monitor 来监测其它变量。
  • 淡橙色部分是FPGA外部的硬件电路也就是电机、电机驱动板、角度传感器这些东西。

另外,除了 fpga_top.sv 中调用的 altpll 原语外,该库的所有代码都使用纯 RTL 编写可以轻易地移植到其它厂商Xilinx、Lattice等的 FPGA 上。

RTL仿真

因为我并没有电机的 Verilog 模型,没法对整个 FOC 算法进行仿真,所以只对 FOC 中的部分模块进行了仿真。

仿真相关的文件都在 SIM 文件夹里,其中包括文件:

文件名 功能
tb_clark_park_tr.sv 对 clark_tr.sv clark变换 和 park_tr.sv park变换的仿真程序
tb_clark_park_tr_run_iverilog.bat 用 iverilog 运行 tb_clark_park_tr.sv 的命令脚本
tb_svpwm.sv 对 cartesian2polar.sv 和 svpwm.sv 和 park_tr.sv 的仿真程序
tb_svpwm_run_iverilog.bat 用 iverilog 运行 tb_svpwm.sv 的命令脚本

使用 iverilog 仿真前,需要安装 iverilog ,见:iverilog_usage

clark_tr和park_tr的仿真

我们先来运行 clark 变换和 park 变换的仿真。

双击 tb_clark_park_tr_run_iverilog.bat 可以直接运行仿真,运行完后会生成波形文件 dump.vcd 。请用 gtkwave 打开 dump.vcd ,导入图4中的这些信号,可以看到如图4的波形(需要你把这些信号改成模拟信号显示的形式,才能看得到图4这种效果:第一步,对于 Verilog 代码中声明为 signed 的信号有符号数需要右键该信号→Data Format→Signed Decimal。第二步右键该信号→Data Format→Analog→Step ,即可把它变成如图的模拟信号的样子)。

对该图4波形的解读:

  • theta 是一个不断递增的角度0→2π→0→2π→...
  • ia, ib, ic 是用 theta 生成的正弦波,相位各自相差 (2/3)*π (也就是说 ia, ib, ic构成了三相正弦波。
  • 使用 clark 变换把 ia, ib, ic 变换成 ialpha, ibeta ,得到一对正交的正弦波(相位相差 π/2 )。
  • 使用 park 变换把 ialpha, ibeta 变换到定子坐标系,得到定值 id 和 iq (因为实际的计算误差,所以得到的是近似的定值,而不是严格的定值)。
图4:对 clark_tr 与 park_tr 仿真的波形。

cartesian2polar和svpwm的仿真

现在来运行 cartesian2polar (直角坐标转极坐标系)和 svpwm 的仿真。

双击 tb_svpwm_run_iverilog.bat 可以直接运行仿真,运行完后会生成波形文件 dump.vcd 。请用 gtkwave 打开 dump.vcd ,可以看到如图5的波形。

注意: pwma_duty、pwmb_duty、pwmc_duty 这三个信号不在顶层,你要在 svpwm 这个模块内才能找到这三个信号。

图5波形的解读:

  • theta 是一个不断递增的角度0→2π→0→2π→...
  • x 和 y 是用 theta 生成的正交的正弦波或者说y是正弦波x是余弦波
  • 把 (x,y) 视作直角坐标值,然后 cartesian2polar 把它转换成极坐标系 (ρ, φ) ,也即 (rho, phi)
  • 把 (rho, phi) 输给 svpwm ,产生了 pwma_duty、pwmb_duty、pwmc_duty 这三个马鞍波。如果你熟悉七段式 SVPWM 的原理,就应该知道为什么是马鞍波,这里不做赘述。
  • pwma_duty、pwmb_duty、pwmc_duty 分别决定了 pwm_a, pwm_b, pwm_c 的占空比duty这个单词就是占空比的意思
图5:对 cartesian2polar 与 svpwm 仿真的波形。

放大波形,可以看到确实是 duty 值越大,对应的 pwm 信号的占空比就越大,如图6

图6图5的放大。

FAQ

一些用户在读代码时向本人反馈了一些疑问,我将它们整理如下。

关于 PCB 设计

  • 相电流低压侧3电阻采样需要运放放大电流信号才能作为AD输入但你开源的PCB似乎找不到运放放大
  • 开源的PCB使用了一体化的三相无刷电机驱动芯片MP6540U13相电流采样电阻和放大器是集成在里面的。请注意MP6540的 SOA, SOB, SOC 三个引脚,它们就是三相电流放大后的电压输出。

关于 FOC 中的数学计算

  • 三电阻采样后的电流重构的那部分代码如何理解?为什么 ia = ADCb + ADCc - 2*ADCa

  • 我们知道电机的相电流 ia, ib, ic 是双极性的即有正有负的分别代表流出电机和流入电机但我们常用的ADC往往都是单极性的即只能采样正电压。那么相电流采样-放大电路就必须考虑双极性到单极性的转换问题工程上用的方法往往都是反向放大加偏置包括MP6540内置采样-放大电路也是这种方案它输出给ADC的电压遵循公式 ADCa = -R × ia + Voff ADCb = -R × ib + Voff ADCc = -R × ic + Voff 其中 R 是放大系数R>0也叫跨阻放大系数因为 R 的量纲和电阻一样因为反向放大所以加了负号。Voff 是偏置电压,以此保证 ADCa, ADCb, ADCc 是单极性的。另外又有基尔霍夫电流定律KCL ia + ib + ic = 0 联立以上公式,推出: 3 * Voff = ADCa + ADCb + ADCc 令 R = 1/(3*k) ,于是: ia = (Voff - ADCa) / R = k × (3 * Voff - 3 × ADCa) = k × (ADCa + ADCb + ADCc - 3 × ADCa) = k × (ADCb + ADCc - 2×ADCa) 于是就有了你问的 ia = ADCb + ADCc - 2×ADCa 。 你肯定会疑惑,系数 k 哪儿去了这个并不在乎因为整个FOC都是线性系统通过调整PID参数能跑就行系数只在理论分析时有用。

  • : 你的程序里是不是都没有在乎系数K包括clark变换中得到的Iα和Iβ和ia、ib、ic也并没有满足严格的公式关系程序中得到的Iα和Iβ是理论值的2倍。这是不是也可以用调整PID参数的思想来解释

  • 是的,我很多地方的代码也与理论公式的系数不同,比如 clark 变换公式本来是: Iα = Ia - Ib/2 - Ic/2 Iβ = √3/2 * (Ib - Ic) 我多乘了个2 Iα = 2 * Ia - Ib - Ic Iβ = √3 * (Ib - Ic) 这是出于避免整数计算的截断导致的数据位丢失,比如 Ib/2 就会让 Ib 的最低 bit 丢失。不过实际上这种小误差基本不会影响控制质量。这种系数问题可以通过 PID 调参来消除。

  • : 你的代码中的 cartesian2polar.sv 是把电压矢量从转子直角坐标系 (Vd, Vq) 变换到转子极坐标系 (Vrρ, Vrθ),目的是什么?

  • 书上一般会说 SVPWM 模块输入的是定子直角坐标系下的电压,但我实现的 SVPWM 输入的是定子极坐标系下的电压,两种方法在数学上是等价的。而且 SVPWM 在 FPGA 里用查找表(ROM)实现,因此两种方法对电路复杂度影响不大。另外,输入极坐标系的 SVPWM 还带来 2 个好处和 1 个代价,好处 1 是更方便在开发过程中让电机开环地转起来(只需要让角度递增即可)。好处 2 是极坐标系下的 park 变换更简单,只需要用电压在转子坐标系中的角度减去电角度就能得到电压在定子坐标系中的角度。 代价是需要在 PID 的后面、park 变换的前面实现一个直角坐标系转极坐标系的运算,即 cartesian2polar.sv

关于 ADC 采样时机

  • AD7928只有一个T/H(采样保持器)也就意味着这个AD一次只能保持一个通道的数据也就是说当通道1 打开的时候采集A相的电流然后采集完再打开通道2采集B相的电流那么这并不是一个同步采集的过程如何实现AD采样三相电流的同步输出

  • 在电流采样问题上,我们与大多数 FOC 方案相同,因为 MP6540 里的采样电阻在下桥臂,所以规定 SVPWM 在每个控制周期(约 55us对应频率18kHz内至少有一段时间里 3 相都是下通上闭称为采样窗口。FOC的基础振幅也就是 foc_top.sv 里的 MAX_AMP 参数)越大,采样窗口越短。当 MAX_AMP=9'd511 (最大值) 时,采样窗口长度就是 0 了。而默认的 MAX_AMP=9'd384 大概会让采样窗口长度为十几us。 AD7928 只有一个T/H所以采样窗口内要做 3 次采样,这个过程是 hold_detect.sv 和 adc_ad7928.sv 共同控制的hold_detect.sv负责在采样窗口开始时延迟一段时间来让电流趋于稳定后发出 sn_adc 信号脉冲,来告诉 adc_ad7928.sv 可以开始工作了(可以通过 foc_top.sv 里的 SAMPLE_DELAY 来调整该延迟)。然后 adc_ad7928.sv 内部就会自动串行地完成 3 个通道的采样最后再同步提交3个采样结果提交的同时产生 o_en_adc 信号脉冲。注意adc_ad7928.sv是一个通用的 AD7928 控制器,每收到一个 i_sn_adc 信号脉冲,就进行一系列串行采样。其中采样多少次,每次采样哪一个通道,都可以通过 adc_ad7928.sv 里的 parameter 来配置。所有通道采样结束后产生o_en_adc信号脉冲同时同步提交所有通道的结果。 我把 foc_top.sv 用来连接 ADC控制器的接口也就是sn_adc, en_adc, adc_a, adc_b, adc_c 这几个信号设计成同步读入3通道数据是出于通用性、简约性和可移植性的原则考虑因为这样的同步读入接口是 ADC 的一种高度抽象最容易让人理解其时序sn_adc脉冲命令ADC控制器开始工作。en_adc指示ADC读取器结束工作同时 adc_a, adc_b, adc_c 上产生结果)。如果用户用了其它 ADC 型号,只需按照这个时序的抽象来具象地编写 ADC 控制器即可,而 foc_top.sv 并不关心拟用的是1个串行的ADC还是3个并行的ADC反正你都要给我同步提交。当然用户必须自己算好 hold_detect.sv 的延时 + ADC 采样三个通道(也即 sn_adc 脉冲和 en_adc 脉冲的时间差) 是小于采样窗口的长度的。

  • 进行串行采样的时候不会出现这样的问题第一个时钟周期采到的是A相的电流第2个时钟周期采的是B相第3个时钟周期采到的是C相而我们知道相电流是正弦变化的这三个时钟周期采的A B C 三相电流并不是同一时刻的相电流,因此这将产生误差,而时钟周期是很短的,是不是产生的误差几乎可以忽略不计呢?

  • 首先指正一个不准确的地方, AD7928 的接口是 SPI (一种串行接口),其采样是多个时钟周期(而不是一个时钟周期)内完成的。因此三相的采样间隔是几十个时钟周期(具体数字我忘了,你可以仿真确定一下)。 虽然不同步但3相的采样毕竟都是在同一个控制周期的采样窗口几微秒内。而相电流的变化通常以控制周期即几十微秒为尺度变化例如 3000r/min50r/s的转子若极对数=7其相电流是350Hz周期28ms的正弦波其变化在几微秒内是可以忽略的。

参考资料

Languages
Verilog 99.9%
Batchfile 0.1%