深圳市航顺芯片技术研发有限公司
直播中

小芳

13年用户 960经验值
私信 关注
[问答]

如何使用FPGA控制其输出正弦波?

如何使用FPGA控制其输出正弦波?

回帖(1)

叶冬琳

2021-10-26 14:39:16
本次介绍ADI公司的数字频率合成(DDS)芯片——AD9850以及如何使用FPGA控制其输出正弦波。
本文分三个部分讲解:一是AD9850芯片介绍,包括基本引脚功能、驱动时序以及相关注意事项;二是基于FPGA的AD9850的驱动的建模;三是仿真和调试结果。

PART1:AD9850认知
1、模式选择:并行和串行两种。
2、通信协议:实际上属于SPI协议的变种。串行时很好理解,并行时可以理解为在时钟的上升沿将整个字节的数据并行采集,这里将这种spi模式定义为并行spi。
3、寄存器配置:内部包含两个两个寄存器,相位控制寄存器(8bit)和频率控制寄存器(32bit,可分成4个8bit寄存器),总40bit,分为5个控制字写入,从高位到低位每8bit为一个控制字寄存器,分别为W0、W1、W2、W3、W4。相位寄存器(命为W0),其中W0_bit2-bit0=000(无论串并模式,手册中其他组合定义为厂家模式,慎用),实际上,配置W0=8’h00(0°相位)即可,高5位为初相位配置数据位。
W1-W4共4个控制字组成32bit的频率寄存器。故fo=reg_dds*f_osc/2^32.(单位为MHz)
3、并行模式,先后写入的控制字顺序为W0-W1----W4。
4、极限输出:
理论上,fo_max=1/2*f_osc,取f_osc=125MHz时,fo_max=62.5MHz,实际上输出40MHz(芯片手册说的输出f不超过0.33*fosc,125MHz时约41.25MHz)是没有问题的。频率越高,输出的信号幅值越低(正常的,这是因为输出呈sinc(Sa函数)规律)。
4、芯片正常工作时,5V、125MHz晶振,Pidss=380mW(典型值),实际上这个时候芯片是比较烫的,属于正常现象,也符合电流型输出特点(并不像芯片吹的低功耗)。
5、方波输出(可以调节占空比)功能,正常时需要调节模块的滑动变阻器(修改比较器的门限值),虽然手册上声称内部含有高速比较器,但实际上最高输出方波的频率理论上不超过1MHz(内部比较器的输出带宽有限,导致方波的告辞谐波丢失严重,进而出现失真),如果想获得高频的方波,可以采用TLV3502代替内部比较器,这块片子输出50MHz方波是没有问题的。
NOTE:方波测量时最好不开启带宽限制(特别是有些示波器具有数字滤波功能);dds的测量最好使用频谱分析仪等仪器测试(有条件的选手)。
6、通信协议的时序要求(见下图)
【1】主威廉希尔官方网站 复位





【2】数据写入(并行)





【3】总体时序参数





以上是AD9850的相关介绍,下面介绍FPGA建模。

PART2:fpga建模
Step1:spi主体建模
由前所述,特提出并行spi,值得是在时钟边沿写入一个字节数据,谓之并行,很明显这种形式的spi具有较高的传输效率和速率。这样,结合之前写过的spi时序,进行简单的修改便可完成主体通信时序的建模。具体修改如下:定义并行接口(只需要将sda设置成并行的端口,位数为8)。

//AD9850并行数据
always @ (posedge clk, negedge rst_n) begin
        if (!rst_n)
                sda <= {D0_WID{1'b0}};
        else if ((state == WRITE || state == START) && sclk_cnt == 4'd0)
                sda <= sda_buf;
end

always @ (*) begin
        case (bit_cnt)
                5'd0:         sda_buf = sda_buf1[7 : 0];
                5'd4:         sda_buf = sda_buf1[15 : 8];
                5'd3:         sda_buf = sda_buf1[23 : 16];
                5'd2:         sda_buf = sda_buf1[31 : 24];
                5'd1:         sda_buf = sda_buf1[39 : 32];
                default:         sda_buf = {D0_WID{1'b0}};
        endcase
end
endcaseend NOTE:本次设计中的40bit的寄存器数据,其中第一个自己数据W0放在了sda_buf1的低8位,实际上,最好将其放在最高位,那么,上述代码的cade语句中,bit_cnt==5'd0时,就应该赋值39:32,其余的按照规律输出即可。
此外,还需要加上一个复位输出功能,如上,模块含有整体复位功能,实际上可以使用fpga主板的复位引脚给之复位(注意,模块复位是高电平,fpga是低电平)。这里,简单的采用一个计数器完成该部分功能,具体不做展示。

Step2:数据供给模块建模
简单说,就是给AD9850的40bit的寄存器的值,即遗传01代码,作为验证程序,这里从简即可。具体的思路是,通过fpga主板上的按键外设作为系统输入,判断按键的输入键值,进行相应译码,得到对应的寄存器目标配置值。本次设计设定了5中输出频率,对应关系如下:
表1 频率值对应16进制数值

频率输出(MHz)
1KHz
1
10
20
40
50
16进制值
00008637
020c49ba
147ae147
28f5c28f
51eb851e
66666666
由于采用了按键,最好加上按键消抖,此模块之前已经做了介绍(代码资料可在我的主页下载,博客中有相关模块的说明)。

Step3:仿真和调试结果
简单仿真,得到的仿真结果基本符合时序要求。以下直接给出调试结果:
图1:1KHz输出





图2:1MHz输出





图3:10MHz输出





图4:20MHz输出





图5:40MHz输出





图6:50MHz输出





通过简单的按键输入,即可使得输出在以上几个频点值之间进行切换输出,且在示波器观察到的波形效果较好,知本次建模完成了相应功能。

最后,习惯给出整体的RTL功能视图:


举报

更多回帖

发帖
×
20
完善资料,
赚取积分