这些小活动你都参加了吗?快来围观一下吧!>>
电子产品世界 » 论坛首页 » DIY与开源设计 » 电子DIY » 基于Si5351A一款可产生8KHz-160MHz高频信号方波的频率产生器DIY

共4条 1/1 1 跳转至

基于Si5351A一款可产生8KHz-160MHz高频信号方波的频率产生器DIY

工程师
2020-08-31 21:48:53     打赏

如下是一款可产生 8KHz-160MHz 高频信号方波的频率产生器,采用Si5351A 芯片实现,输出的频率信号,可以通过I2C总线对其进行控制。电路还是很简单的,适合大家DIY哦!


image.png

驱动源码.c如下:

/********************************************

主控芯片:STM32F405RGT6主频168Mhz

晶体频率:HSE=8Mhz  SYSCLK=168Mhz

通讯方式:I2C

函数功能:Si5351时钟芯片


********************************************/

#include "si5351a.h"

#include "delay.h"


void IICstart()//IIC总线起始信号

{

  SDA(1);

  CLK(1);

  SDA(0);

    delay_us(100); 

  CLK(0);

    delay_us(100);

}

void IICstop()//IIC总线停止信号

{

  CLK(0);

    delay_us(100); 

  SDA(0);

    delay_us(100); 

  CLK(1);

    delay_us(100); 

  SDA(1);

    delay_us(100); 

}

void IICsend(u8 DATA)//IIC总线发送信数据

{

u16 i;

    for(i=0;i<8;i++) //发送一个字节数据 

    {     

        CLK(0); //拉低时钟线,准备开始给SDA赋值

        if((DATA&0x80)==0)  

        {SDA(0);}  

        else  

        {SDA(1);}  

        DATA<<=1;  

        delay_us(100);    

        CLK(1);

        delay_us(100);      //等待从设备把数据接收完

}  

    CLK(0);  

    SDA(1);    //释放数据线   

    delay_us(100);   

    CLK(1); 

    delay_us(100); 

while(i<1000){i++;}//等待从设备应答ACK 

CLK(0); //释放时钟线,为下次操作做准备

}

void IICsendreg(uint8_t reg, uint8_t data)

{

  IICstart();      //起始信号

  delay_us(200);  

  IICsend(0xC0);   //发送设备地址+写信号

  delay_us(200);  

  IICsend(reg);    //内部寄存器地址

  delay_us(200);  

  IICsend(data);   //内部寄存器数据

  delay_us(200);  

  IICstop(); 

}

void Si5351Init()//初始化Si5351芯片

{

GPIO_InitTypeDef GPIO_Initure;

    __HAL_RCC_GPIOB_CLK_ENABLE(); //开启GPIOB时钟

GPIO_Initure.Pin=GPIO_PIN_10|GPIO_PIN_11;//PB10,11引脚

GPIO_Initure.Mode=GPIO_MODE_OUTPUT_PP;//复用推挽输出

GPIO_Initure.Pull=GPIO_PULLUP;//上拉模式

GPIO_Initure.Speed=GPIO_SPEED_HIGH;//GPIO速度高速

  HAL_GPIO_Init(GPIOB,&GPIO_Initure);//初始化GPIOB

}

void SetFrequency(uint32_t frequency)//设置时钟频率

{

  uint32_t pllFreq;

  uint32_t xtalFreq = XTAL_FREQ;// 晶体频率

  uint32_t l;

  float f;

  uint8_t mult;

  uint32_t num;

  uint32_t denom;

  uint32_t divider;

  divider = 900000000 / frequency; //锁相环频率:900 mhz

  if (divider % 2) divider--; //确保一个更整数除法比率

  pllFreq = divider * frequency; //计算pllFrequency:分频器*所需的输出频率

  mult = pllFreq / xtalFreq; //确定所需的pllFrequency的乘数

  l = pllFreq % xtalFreq; //它有三个部分:

  f = l; //乘是一个整数,必须在15 . . 90

  f *= 1048575; //num和分母项是小数部分,分子和分母

  f /= xtalFreq; //每20位(范围0 . . 1048575)

  num = f; //实际的乘数是乘+ num /分母项

  denom = 1048575; //为简单起见我们将分母最大1048575

//设置锁相环与倍增系数计算

  SetPLLClk(SI_SYNTH_PLL_A, mult, num, denom);

//设置MultiSynth分配器0分计算。

//最后R分裂阶段可以除以2的幂,从1 . . 128。

//由常量SI_R_DIV1 reprented SI_R_DIV128(见si5351a。h头文件)

//如果你想在1兆赫兹以下输出,你必须使用

//最后R分裂阶段


  SetMultisynth(SI_SYNTH_MS_0,divider,SI_R_DIV_1);

//重置锁相环。这将导致输出的故障。对于小的变化

//参数,不需要复位锁相环,没有故障


  IICsendreg(SI_PLL_RESET,0xA0);

//最后打开CLK0输出(0 x4f)

//并设置MultiSynth0是锁相环的输入

  IICsendreg(SI_CLK0_CONTROL, 0x4F|SI_CLK_SRC_PLL_A);

}

void SetPLLClk(uint8_t pll, uint8_t mult, uint32_t num, uint32_t denom)//设置PPL时钟

{

  uint32_t P1; // PLL config register P1

  uint32_t P2; // PLL config register P2

  uint32_t P3; // PLL config register P3


  P1 = (uint32_t)(128 * ((float)num / (float)denom));

  P1 = (uint32_t)(128 * (uint32_t)(mult) + P1 - 512);

  P2 = (uint32_t)(128 * ((float)num / (float)denom));

  P2 = (uint32_t)(128 * num - denom * P2);

  P3 = denom;


  IICsendreg(pll + 0, (P3 & 0x0000FF00) >> 8);

  IICsendreg(pll + 1, (P3 & 0x000000FF));

  IICsendreg(pll + 2, (P1 & 0x00030000) >> 16);

  IICsendreg(pll + 3, (P1 & 0x0000FF00) >> 8);

  IICsendreg(pll + 4, (P1 & 0x000000FF));

  IICsendreg(pll + 5, ((P3 & 0x000F0000) >> 12) | ((P2 & 0x000F0000) >> 16));

  IICsendreg(pll + 6, (P2 & 0x0000FF00) >> 8);

  IICsendreg(pll + 7, (P2 & 0x000000FF));

}

void SetMultisynth(uint8_t synth,uint32_t divider,uint8_t rDiv)//设置多synth

{

  uint32_t P1; // Synth config register P1

  uint32_t P2; // Synth config register P2

  uint32_t P3; // Synth config register P3


  P1 = 128 * divider - 512;

  P2 = 0; // P2 = 0, P3 = 1 forces an integer value for the divider

  P3 = 1;


  IICsendreg(synth + 0,   (P3 & 0x0000FF00) >> 8);

  IICsendreg(synth + 1,   (P3 & 0x000000FF));

  IICsendreg(synth + 2,   ((P1 & 0x00030000) >> 16) | rDiv);

  IICsendreg(synth + 3,   (P1 & 0x0000FF00) >> 8);

  IICsendreg(synth + 4,   (P1 & 0x000000FF));

  IICsendreg(synth + 5,   ((P3 & 0x000F0000) >> 12) | ((P2 & 0x000F0000) >> 16));

  IICsendreg(synth + 6,   (P2 & 0x0000FF00) >> 8);

  IICsendreg(synth + 7,   (P2 & 0x000000FF));

}


对应的.h头文件如下:

#ifndef _si5351a_h

#define _si5351a_h

#include "system.h"

//Si5351寄存器声明

#define SI_CLK0_CONTROL 16 // Register definitions

#define SI_CLK1_CONTROL 17

#define SI_CLK2_CONTROL 18

#define SI_SYNTH_PLL_A 26

#define SI_SYNTH_PLL_B 34

#define SI_SYNTH_MS_0 42

#define SI_SYNTH_MS_1 50

#define SI_SYNTH_MS_2 58

#define SI_PLL_RESET 177


#define SI_R_DIV_1 0x00 // R-division ratio definitions

#define SI_R_DIV_2 0b00010000

#define SI_R_DIV_4 0b00100000

#define SI_R_DIV_8 0b00110000

#define SI_R_DIV_16 0b01000000

#define SI_R_DIV_32 0b01010000

#define SI_R_DIV_64 0b01100000

#define SI_R_DIV_128 0b01110000


#define SI_CLK_SRC_PLL_A 0x00

#define SI_CLK_SRC_PLL_B 0b00100000

#define XTAL_FREQ 25000000 // Crystal frequency

//IIC总线引脚配置

#define SDA(n) {n?HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_SET):HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_RESET);}

#define CLK(n) {n?HAL_GPIO_WritePin(GPIOB,GPIO_PIN_11,GPIO_PIN_SET):HAL_GPIO_WritePin(GPIOB,GPIO_PIN_11,GPIO_PIN_RESET);}

//相关函数声明

void Si5351Init(void);//初始化Si5351的GPIO

void SetPLLClk(uint8_t pll, uint8_t mult, uint32_t num, uint32_t denom);//设置PPL时钟

void SetFrequency(uint32_t frequency);//时钟Si5351时钟频率

void SetMultisynth(uint8_t synth,uint32_t divider,uint8_t rDiv);//设置多synth

#endif





关键词: Si5351A     高频信号     方波     频率发生器    

工程师
2020-08-31 22:01:25     打赏
2楼

软硬件都有啊,不错!


工程师
2020-08-31 22:06:23     打赏
3楼

I2C 模拟程序的软件~


工程师
2020-08-31 22:10:04     打赏
4楼

资料很丰富嘛!可以学习一番了!


共4条 1/1 1 跳转至

回复

匿名不能发帖!请先 [ 登陆 注册 ]