矿石收音机论坛

 找回密码
 加入会员

QQ登录

只需一步,快速开始

搜索
查看: 2128|回复: 6

AD9833问题请教,用下面的程序,只能输出正弦波,不能输出三角波

[复制链接]
     
发表于 2021-5-15 17:58:29 来自手机 | 显示全部楼层 |阅读模式
程序如下:
sbit f_sync = P2^2;
sbit s_clk         = P2^1;
sbit s_data = P2^0;


#define DDS_SCLK_UP         s_clk=1
#define DDS_SCLK_DOWN         s_clk=0

#define DDS_SDATA_UP         s_data=1
#define DDS_SDATA_DOWN         s_data=0

#define DDS_FSYNC_UP         f_sync=1
#define DDS_FSYNC_DOWN         f_sync=0

//#define BITB 0x0800


//AD9833输入时钟值
unsigned long int MCLK=25000000;      
double p=3.141592653589793;
//寄存器配置数组
unsigned int Config_Data[8];

//频率
unsigned long int code freq[5]={30000,60000,120000,480000,960000};

//函数声明
static void delay2us(unsigned char i);
static void changeFreq(unsigned char key);
static unsigned char getKeyValue();
void writeDDS2Byte(unsigned int config);
void waveGenerate(unsigned long int frequency,double phase,unsigned char signal_type);


//void main(void)
//{
//        waveGenerate(freq[4],2*p,0); //
//}


/*******************************************************************
*        功能:波形发生器
*
*        入口参数:         frequency: 期望得到的信号频率 (frequency<MCLK)
*
*                          phase:期望得到的相位
*
*             signal_type:0正弦波1(三角波),2(方波)   
*
*        默认配置:         0相移,方波不分频
*******************************************************************/
void waveGenerate(unsigned long int frequency,double phase,unsigned char signal_type)
{
        unsigned char k;
        unsigned long int freq_temp;
        unsigned int phase_temp;
        if(frequency>MCLK)
        frequency=MCLK;
        switch(signal_type)
        {
                case 0://正弦波
                        Config_Data[0]=0x2108;//控制寄存器配置值,复位片内其他寄存器,AD9833上电时,期间应复位,要使AD9833复位应将reset位置1,即bit8置1,要通过Vout引脚提供正弦波输出,应将mode(D1)bit清0并将OPBITEN(D5)bit清0,当DIV2=1时即bit3=1,DAC的MSB被直接送至Vout引脚
                        Config_Data[7]=0x2008;//控制寄存器配置值,不复位片内其他寄存器,要使AD9833退出复位,应将该位清零,即bit8位清零,要通过Vout引脚提供正弦波输出,应将mode(D1)bit清0并将OPBITEN(D5)bit清0,当DIV2=1时即bit3=1,DAC的MSB被直接送至Vout引脚
                break;
                case 1://三角波
                        Config_Data[0]=0x210A;//控制寄存器配置值,复位片内其他寄存器,AD9833上电时,期间应复位,要使AD9833复位应将reset位置1,即bit8置1,要通过Vout引脚提供三角波输出,应将mode(D1)1,DIV2=1时即bit3=1,DAC的MSB被直接送至Vout引脚
                        Config_Data[7]=0x200A;//控制寄存器配置值,不复位片内其他寄存器,要使AD9833退出复位,应将该位清零,即bit8位清零,要通过Vout引脚提供三角波输出,应将mode(D1)置1,DIV2=1时即bit3=1,DAC的MSB被直接送至Vout引脚
                break;
                case 2://方波,不分频
                        Config_Data[0]=0x2128;//控制寄存器配置值,复位片内其他寄存器,AD9833上电时,期间应复位,要使AD9833复位应将reset位置1,即bit8置1,当OPBITEN(D5)置1,mode(D1)清0,DIV2=1时即bit3=1,Vout输出DAC数据MSB
                        Config_Data[7]=0x2028;//控制寄存器配置值,不复位片内其他寄存器,要使AD9833退出复位,应将该位清零,即bit8位清零,当OPBITEN(D5)置1,mode(D1)清0,DIV2=1时即bit3=1,Vout输出DAC数据MSB
                        break;
                default://正弦波
                        Config_Data[0]=0x2108;
                        Config_Data[7]=0x2008;
                break;
        }
        //freq_temp=frequency*(2^28/MCLK),MCLK=6Mhz?,2^28/MCLK约等于44.739242666666669
        //phase_temp=phase*(4096/2p),p=3.141592653589793
        freq_temp=frequency*10.73741824; //载入所选频率寄存器的值,此信号会经过如下相位偏移处理,2?/4096*PHASEREG,PHASEREG是所选相位输出频率和参考时钟频率之间的关系,必须考虑所选输出频率和参考时钟频率之间的关系,以免产生不良的输出异常
        phase_temp=phase*(4096/2*p);//651.898646904403295309
        Config_Data[1]=freq_temp&0x3fff;//先将bit15,bit14位清0,控制字写入
        Config_Data[3]=Config_Data[1];//先将bit15,bit14位清0,控制字写入
        Config_Data[2]=(freq_temp&0x0fffc000)>>14;//先将bit15,bit14位清0,控制字写入
        Config_Data[4]=Config_Data[2];//先将bit15,bit14位清0,控制字写入
        Config_Data[5]=phase_temp&0x1fff;
        Config_Data[5]=Config_Data[6];


        Config_Data[1]=Config_Data[1]|0x4000;//(频率寄存器)FREQ0 14 LSBs,bit15=0,bit14=1,FREQ0寄存器写入
        Config_Data[2]=Config_Data[2]|0x4000;//(频率寄存器)FREQ0 14 MSBs,bit15=0,bit14=1,FREQ0寄存器写入
        Config_Data[3]=Config_Data[3]|0x8000;//(频率寄存器)FREQ1 14 LSBs,bit15=1,bit14=0,FREQ1寄存器写入
        Config_Data[4]=Config_Data[4]|0x8000;//(频率寄存器)FREQ1 14 MSBs,bit15=1,bit14=0,FREQ1寄存器写入
//        Config_Data[5]=0xC000;//(相位寄存器)PHASE0
//        Config_Data[6]=0xE000;//(相位寄存器)PHASE1
        Config_Data[5]=Config_Data[5]|0xC000;//(相位寄存器)PHASE0,bit15=1,bit14=1,bit13=0,PHASE0寄存器写入
        Config_Data[6]=Config_Data[6]|0xE000;//(相位寄存器)PHASE1,bit15=1,bit14=1,bit13=1,PHASE1寄存器写入
        for(k=0;k<8;k++)
        {
                writeDDS2Byte(Config_Data[k]);
        }
}

/***********************************************************************************************
*
*        软件模拟SPI
*
*        向AD9833写入数据或控制信息时0,FSYNC应处于低电平并保持低电平,直到数据的16个位均已写入AD9833为止,FSYNC信号以帧方式传输要载入AD9833的16位信息
*
*        向AD9833发送数据时,FSYNC被拉低,单片机以8位字节传输数据,因此每个周期中只有8个SCLK下降沿,要向AD9833中载入剩余的8个位,FSYNC应在第一批8个位传输完成后保持低电平,同时启动第二次写操作来传输数
*        据的第二个字节,第二次写操作结束后FSYNC被拉高,SCLK应在两次写操作之间处于高电平空闲状态
*
************************************************************************************************/
void writeDDS2Byte(unsigned int config)
{
        unsigned char i;
        DDS_SCLK_DOWN;//拉低SCLK
        _nop_();//延时
        DDS_FSYNC_UP;//拉高FSYNC
        _nop_();//延时
        DDS_SCLK_UP;//拉高SCLK
        delay2us(1);//延时1us
        DDS_FSYNC_DOWN;//发送数据时,FSYNC拉低
        _nop_();
        for (i=0; i<16; i++)
        {
                if (config & 0x8000)
                DDS_SDATA_UP;//SDATA拉高
                else
                DDS_SDATA_DOWN;//SDATA拉低
                delay2us(1);//延时1us
                DDS_SCLK_DOWN;//SCLK拉低
                delay2us(1);//延时1us
                DDS_SCLK_UP;//SCLK拉高
                config <<= 1;//config循环左移一位
        }
        DDS_FSYNC_UP;//FSYNC拉高
        _nop_();//延时
        DDS_SCLK_DOWN;//SCLK拉低
        delay2us(100);//延时100us
}


static void delay2us(unsigned char i)
{
        while (--i);
}
     
发表于 2021-5-16 11:46:30 | 显示全部楼层
输出有没有低通滤波器?在滤波器前还是后测的?
回复 支持 反对

使用道具 举报

     
 楼主| 发表于 2021-5-17 06:44:52 来自手机 | 显示全部楼层
mov99 发表于 2021-5-16 11:46
输出有没有低通滤波器?在滤波器前还是后测的?

30M有源晶振输出这样的波形怎么不是方波
IMG_20210516_185422.jpg
回复 支持 反对

使用道具 举报

     
发表于 2021-5-17 10:37:35 | 显示全部楼层
haohongshuai 发表于 2021-5-17 06:44
30M有源晶振输出这样的波形怎么不是方波

示波器带宽不够,显示失真而已
回复 支持 反对

使用道具 举报

     
 楼主| 发表于 2021-5-17 12:00:38 来自手机 | 显示全部楼层
iffi123 发表于 2021-5-17 10:37
示波器带宽不够,显示失真而已

带宽是200M的,
回复 支持 反对

使用道具 举报

     
发表于 2021-5-17 12:16:31 | 显示全部楼层

对方波的奇数谐波来说,200M还是不够
回复 支持 反对

使用道具 举报

     
发表于 2021-5-17 17:08:02 | 显示全部楼层
本帖最后由 mov99 于 2021-5-17 18:12 编辑
haohongshuai 发表于 2021-5-17 06:44
30M有源晶振输出这样的波形怎么不是方波


30M有源晶振不是信号源,是时钟信号,不是纯正的方波也不奇怪。另外,也许是谐波丰富还有和测试方法有关,开启FFT功能看下,虽然不能定量参考的意义还是有的。200MHz的示波器看30MHz的方波没问题。

常用示波器探头1:1档的带宽一般6MHz,稍好的也不过10MHz,想达到探头标称带宽应该置10:1档;示波器本身自带探头补偿校准信号,校准后在用。测方波用DC档应该好些。

个人观点,仅供参考。
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 加入会员

本版积分规则

小黑屋|手机版|矿石收音机 ( 蒙ICP备05000029号-1 )

蒙公网安备 15040402000005号

GMT+8, 2025-4-28 07:53

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表