草庐IT

【基础知识】~ 分频器

IC二舅 2023-05-14 原文

本章目录:

1. 什么是分频

分频就是生成一个新时钟,该新时钟的频率是原有时钟频率的整数分之一倍,新周期是原有周期的整数倍。再简单来说,让你手撕一个四分频电路,就是写代码生成一个周期是原来四倍的时钟,如果手撕一个三分频电路,就是写代码生成一个周期是原来三倍的时钟。但是奇数分频会比偶数分频复杂一些。

2. 偶数分频

2.1 怎么实现二分频呢?

检测参考时钟,每一个上升沿到来时,新的时钟翻转一次。

2.2 代码实现:

module div_2(
		input clk,
		input rst_n,
		output reg clk_out	
);

parameter n_div = 2;
reg [2:0] cnt;

always @ (posedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out <= 1'b0;
		cnt <= 3'b0;
	end
	else begin
		if(cnt == n_div/2 - 1) begin
			clk_out <= ~clk_out;
			cnt <= 3'b0;
		end
		else begin
			cnt <= cnt + 1'b1;
		end
	end
end 
endmodule

2.3 testbench文件

`timescale 1 ns/ 1 ps


module div_2_tb();

	reg clk;
	reg rst_n;
	// wires                                               
	wire clk_out;

	div_2 i1 (
		.clk(clk),
		.clk_out(clk_out),
		.rst_n(rst_n)
	);

	initial                                                
		begin                                                                      
			clk <= 0;
			rst_n <= 1;
			#50
			rst_n <= 0;
			#50
			rst_n <= 1;
			#50000;
			$stop;                                                              
		end    															
	always                                                                 
		begin                                                                         
			  #10 clk = ~clk;                                                                                                                                      
		end                                                  
endmodule

2.4 仿真波形

搞定!从图中可以看到,clk_out每过两个clk时钟上升沿就flip-flop一次,这样一个clk_out的周期就是clk周期的两倍,也即实现了二分频。

那如果是任意偶数分频呢,这就得数经过了多少上升沿,如果是四分频,那就每过两个时钟上升沿(两周期),新信号flip-flop一次,flip-flop两次为新信号的一个周期,即是周期是原时钟的四倍。同理,8分频就是每数4个上升沿,flip-flop一次,以此类推,可以实现任意偶数分频了。

2.5 笔试面试变式题

占空比分析:我们上面写的任意偶数分频代码的占空比都是50%,实际上面试手撕代码不会让你50%占空比,那怎么办?

答:进行计数,对于一个八分频,开始就把时钟设为高电平,我用cnt 计数到两个时钟上升沿后再把它拉低,计数到7后cnt 拉低 时钟拉高,这样就实现了两个周期高,六个周期低,占空比为2/8 即 25% 的八分频。

参考文献:分频器

3. 奇数分频(除了1分频)

怎么写一个三分频呢?一个占空比不是50%的三分频是好写的,同样利用一个cnt变量对上升沿计数来实现。比如实现占空比为1/3的三分频,那么只要数一个上升沿,输出高电平,数两个上升沿输出低电平就能解决。

3.1 代码实现(3分频,非50%占空比)

module odd_div (
			input clk,
			input rst_n,
			output reg clk_out,
			output reg [1:0] cnt
);

always @ (posedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out <= 1'b0;
		cnt <= 2'b00;
	end
	else begin
		if(cnt == 0) begin
			clk_out <= 1'b0;
			cnt <= cnt + 1'b1;
		end
		else if(cnt == 2) begin
			clk_out <= 1'b1;
			cnt <= 2'b00;
		end
		else begin
			cnt <= cnt + 1'b1;
		end
	end
end 

endmodule

3.2 testbench文件

`timescale 1ns/1ps

module odd_div3_tb ();

reg clk;
reg rst_n;
wire clk_out;
wire [1:0] cnt;

initial begin
		clk <= 0;
		rst_n <= 1;
		#50
		rst_n <= 0;
		#50
		rst_n <= 1;
		#1000
		$stop;
end

always begin
	#10 clk = ~clk;
end

odd_div div3 (
			.clk(clk),
			.rst_n(rst_n),
			.clk_out(clk_out),
			.cnt(cnt)
);

endmodule

3.3 仿真


这里抛出一个小问题,3分频之后为什么有延时呢?
我觉得应该是我用的门级仿真,所以会有一定的延时存在。

3.4 那么如何实现一个占空比50%的三分频呢?

答:将一个占空比1/3 上升沿采样的三分频和一个占空比1/3下降沿采样的三分频结果,做或运算

什么意思?怎么想到这样做的?下面来解答!

任何奇数 2N-1 (除1分频外)分频都可以表示由 N-1个高电平周期和 N个低电平周期组成。(占空比最接近50%,但小于50%)

写两个这样的分频器,一个上升沿采样的分频器——div1,一个下降沿采样的分频器——div2,在相同 cnt 判断切换高低电平的条件下,就一定有 div1 领先(或滞后)于 div2 半个参考周期,这样他们相后会使得新的结果还是 2N-1 分频,但是高电平周期变为 N-1+1/2=N-1/2,低电平周期变为 N-1/2,从而高低电平持续时间相等,实现50%占空比的任意奇数分频器。

3.4.1 代码实现(以50%占空比的七分频举例)

module odd_div7_half (
input clk,
input rst_n,
output reg [2:0] cnt1,
output reg [2:0] cnt2,
output reg clk_out1,
output reg clk_out2,
output clk_out
);

localparam div_cnt = 7;

always @ (posedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out1 <= 1'b0;
		cnt1 <= 3'b000;
	end
	else begin
		if(cnt1 == (div_cnt-1)/2) begin
			clk_out1 <= 1'b1;
			cnt1 <= cnt1 + 1;
		end
		else if(cnt1 == div_cnt-1) begin
			clk_out1 <= 1'b0;
			cnt1 <= 3'b000;
		end
		else begin
			cnt1 <= cnt1 + 1;
		end
	end
end

always @ (negedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out2 <= 1'b0;
		cnt2 <= 3'b000;
	end
	else begin
		if(cnt2 == (div_cnt-1)/2) begin
			clk_out2 <= 1'b1;
			cnt2 <= cnt2 + 1;
		end
		else if(cnt2 == div_cnt-1) begin
			clk_out2 <= 1'b0;
			cnt2 <= 3'b000;
		end
		else begin
			cnt2 <= cnt2 + 1;
		end
	end
end

assign clk_out = clk_out1 | clk_out2;

endmodule

3.4.2 testbench文件

`timescale 1ns/1ps

module div7_half_tb();

	reg clk;
	reg rst_n;
	wire [2:0] cnt1;
	wire [2:0] cnt2;
	wire clk_out1;
	wire clk_out2;
	wire clk_out;

initial begin
	clk <= 0;
	rst_n <= 1;
	#50
	rst_n <= 0;
	#50
	rst_n <= 1;
	#1000
	$stop;
end

always begin
	#10 clk = ~clk;
end

odd_div7_half u0(
		.clk(clk),
		.rst_n(rst_n),
		.cnt1(cnt1),
		.cnt2(cnt2),
		.clk_out1(clk_out1),
		.clk_out2(clk_out2),
		.clk_out(clk_out)
);
endmodule

3.4.3 仿真结果

到此,我们就实现了任意50%占空比的奇数分频器,总结一下。说了那么多我们已经可以独立写出任意合理占空比的偶数分频器,以及任意50%占空比的奇数分频器。但是,如果让手撕一个不是50%占空比的奇数分频器怎么办???

3.5 如何实现一个非常规占空比的奇数分频器

2022数字IC秋招面经,面试被问了一个非常规占空比的奇数分频器,比如 3/10占空比的五分频, 5/18占空比的九分频?怎么做呢。

这个和上面实现50%任意奇数分频器的原理是类似的,但是采用与运算。用占空比为 2/5 上升沿采样的信号和 2/5占空比下降沿采样的信号相与,这样由于下降沿采样信号滞后上升沿采样信号半个参考周期。
所以相与后,占空比就为 2/5 - 1/10 = 3/10 ,示意图如下:

3.5.1 代码实现

module div5_3to10_half(
input clk,
input rst_n,
output reg clk_out1,
output reg clk_out2,
output clk_out,
output reg [2:0] cnt1,
output reg [2:0] cnt2
);

localparam div_cnt = 5;

always @ (posedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out1 <= 1'b0;
		cnt1 <= 3'b0;
	end
	else begin
		if(cnt1==(div_cnt/2)-1) begin
			clk_out1 <= 1'b0;
			cnt1 <= cnt1 + 1;
		end
		else if(cnt1==div_cnt-1) begin
			clk_out1 <= 1'b1;
			cnt1 <= 3'b0;
		end
		else begin
			cnt1 <= cnt1 + 1;
		end
	end
end 

always @ (negedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out2 <= 1'b0;
		cnt2 <= 3'b0;
	end
	else begin
		if(cnt2==(div_cnt/2)-1) begin
			clk_out2 <= 1'b0;
			cnt2 <= cnt2 + 1;
		end
		else if(cnt2==div_cnt-1) begin
			clk_out2 <= 1'b1;
			cnt2 <= 3'b0;
		end
		else begin
			cnt2 <= cnt2 + 1;
		end
	end
end 

assign clk_out = clk_out1 & clk_out2;
endmodule

3.5.2 testbench文件

`timescale 1ns/1ps

module div_3to10half_tb();

reg clk;
reg rst_n;
wire clk_out1;
wire clk_out2;
wire clk_out;
wire [2:0] cnt1;
wire [2:0] cnt2;

initial begin
	clk <= 0;
	rst_n <= 1;
	#50
	rst_n <= 0;
	#50
	rst_n <= 1;
	#10000
	$stop;
end

always begin
	#10 clk = ~clk;
end

div5_3to10_half u0(
		.clk(clk),
		.rst_n(rst_n),
		.clk_out1(clk_out1),
		.clk_out2(clk_out2),
		.clk_out(clk_out),
		.cnt1(cnt1),
		.cnt2(cnt2)	
);

endmodule

3.5.3 仿真结果


不论之后遇到让你实现什么样子的奇数分频,具体修改只需要改cnt判断数值以及把clk_out 的赋值从clk_out1,clk_out2相或改成相与。 大功告成!!!

参考文献:2

4. 小数分频

上边讲完了任意占空比任意偶数分频器,以及任意占空比任意奇数分频器后,下面讨论更加夸张的小数分频。什么是小数分频?以及怎么实现 9/4,17/3 ,6.3 等任意小数分频??

编码小数分频,就不能看微观了,要用宏观的眼界去看,比如实现一个 17/3 分频,表达成:17 除以 3 得商为 5 余2

那么我们就可以通过5(商)分频和7(商+余数)分频 来实现 17/3 分频。

现在我们来确定5分频和7分频的次数,设:

5分频的次数为a , 7分频的次数为b;

那么应该有:

    a+b=3(除数)

    5a+7b = 17(被除数)

解得a=2,b=1,也就是说通过2次5分频1次7分频可得到 17/3 分频

这样编程完之后就实现了17/3分频,那么自然会有个疑问,为什么这样就实现了?详细看,仍然是五分频管五分频的,七分频管七分频的,大家各管各的,这算哪门子的小数分频啊??

所以就要提到我们前面说的宏观来看。宏观来看,总共是17个时钟周期,由三个分频器均分,那么平均每个分频器就是分到17/3了,这就是小数分频,这是一个宏观的平均概念。解释完概念,我们下面开始手撕代码。

4.1 代码实现

module div_decimal(
input clk,
input rst_n,
output reg clk_out,
output reg [4:0] cnt,
output reg [2:0] cnt_2to5,
output reg [2:0] cnt_1to7
);

//17/3 小数分频
always @ (posedge clk or negedge rst_n) begin
	if(!rst_n) begin
		clk_out <= 0;
		cnt <= 5'b0;
		cnt_2to5 <= 3'b0;
		cnt_1to7 <= 3'b0;
	end
	else begin
		//2次5分频
		if(cnt<10) begin
			if(cnt_2to5==0) begin
				clk_out <= 0;
				cnt_2to5 <= cnt_2to5 + 1;
			end
			else if(cnt_2to5==4) begin
				clk_out <= 1;
				cnt_2to5 <= 3'b0;
			end
			else begin
				cnt_2to5 <= cnt_2to5 + 1;
			end
			cnt <= cnt + 1;
		end
		//1次7分频
		else if(cnt>9 && cnt<17) begin
			if(cnt_1to7==0) begin
				clk_out <= 0;
				cnt_1to7 <= cnt_1to7 + 1;
			end
			else if(cnt_1to7==6) begin
				clk_out <= 1;
				cnt_1to7 <= 3'b0;
				cnt <= 5'b0;
			end
			else begin
				cnt_1to7 <= cnt_1to7 + 1;
			end
			cnt <= cnt + 1;
		end
	end
end



endmodule

4.2 testbench文件

`timescale 1ns/1ps

module div_decimal_tb();
reg clk;
reg rst_n;
wire clk_out;
wire [4:0] cnt;
wire [2:0] cnt_2to5;
wire [2:0] cnt_1to7;

initial begin
	clk <= 0;
	rst_n <= 1;
	#20
	rst_n <= 0;
	#20
	rst_n <= 1;
	#5000
	$stop;
end

always begin
	#10 clk = ~clk;
end

div_decimal u0(
.clk(clk),
.rst_n(rst_n),
.clk_out(clk_out),
.cnt(cnt),
.cnt_2to5(cnt_2to5),
.cnt_1to7(cnt_1to7)
);
endmodule

4.3 仿真结果


如同我们分析的一样,两个五分频,一个七分频,每十七个周期循环一次。即每十七个周期有三个分频器,平摊下来就是 17/3 。

小数分频的缺点就是 占空比不为50%,要想实现50%占空比的小数分频,涉及很多算法,具体算法十分复杂,一般不会作为手撕代码题。

参考文献:3

若有同学对工程文件感兴趣的,可以后台私信我发送工程文件。如有错误,及时更正,本资料仅提供学习,不可做为商业用途,因为我是一个菜鸡!!!

==========================================================================

声明

本人所有系列的文章,仅供学习,不可商用,如有侵权,请告知,立删!!!

本人主要是记录学习过程,以供自己回头复习,再就是提供给后人参考,不喜勿喷!!!

如果觉得对你有用的话,记得收藏+评论!!!

有关【基础知识】~ 分频器的更多相关文章

  1. postman接口测试工具-基础使用教程 - 2

    1.postman介绍Postman一款非常流行的API调试工具。其实,开发人员用的更多。因为测试人员做接口测试会有更多选择,例如Jmeter、soapUI等。不过,对于开发过程中去调试接口,Postman确实足够的简单方便,而且功能强大。2.下载安装官网地址:https://www.postman.com/下载完成后双击安装吧,安装过程极其简单,无需任何操作3.使用教程这里以百度为例,工具使用简单,填写URL地址即可发送请求,在下方查看响应结果和响应状态码常用方法都有支持请求方法:getpostputdeleteGet、Post、Put与Delete的作用get:请求方法一般是用于数据查询,

  2. 软件测试基础 - 2

    Ⅰ软件测试基础一、软件测试基础理论1、软件测试的必要性所有的产品或者服务上线都需要测试2、测试的发展过程3、什么是软件测试找bug,发现缺陷4、测试的定义使用人工或自动的手段来运行或者测试某个系统的过程。目的在于检测它是否满足规定的需求。弄清预期结果和实际结果的差别。5、测试的目的以最小的人力、物力和时间找出软件中潜在的错误和缺陷6、测试的原则28原则:20%的主要功能要重点测(eg:支付宝的支付功能,其他功能都是次要的)80%的错误存在于20%的代码中7、测试标准8、测试的基本要求功能测试性能测试安全性测试兼容性测试易用性测试外观界面测试可靠性测试二、质量模型衡量一个优秀软件的维度①功能性功

  3. ES基础入门 - 2

    ES一、简介1、ElasticStackES技术栈:ElasticSearch:存数据+搜索;QL;Kibana:Web可视化平台,分析。LogStash:日志收集,Log4j:产生日志;log.info(xxx)。。。。使用场景:metrics:指标监控…2、基本概念Index(索引)动词:保存(插入)名词:类似MySQL数据库,给数据Type(类型)已废弃,以前类似MySQL的表现在用索引对数据分类Document(文档)真正要保存的一个JSON数据{name:"tcx"}二、入门实战{"name":"DESKTOP-1TSVGKG","cluster_name":"elasticsear

  4. ruby - 我怎样才能更好地了解/了解更多关于 Ruby 的知识? - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。我最近开始学习Ruby,这是我的第一门编程语言。我对语法感到满意,并且我已经完成了许多只教授相同基础知识的教程。我已经写了一些小程序(包括我自己的数组排序方法,在有人告诉我谷歌“冒泡排序”之前我认为它非常聪明),但我觉得我需要尝试更大更难的东西来理解更多关于Ruby.关于如何执行此操作的任何想法?

  5. 【网络】-- 网络基础 - 2

    (本文是网络的宏观的概念铺垫)目录计算机网络背景网络发展认识"协议"网络协议初识协议分层OSI七层模型TCP/IP五层(或四层)模型报头以太网碰撞路由器IP地址和MAC地址IP地址与MAC地址总结IP地址MAC地址计算机网络背景网络发展        是最开始先有的计算机,计算机后来因为多项技术的水平升高,逐渐的计算机变的小型化、高效化。后来因为计算机其本身的计算能力比较的快速:独立模式:计算机之间相互独立。    如:有三个人,每个人做的不同的事物,但是是需要协作的完成。    而这三个人所做的事是需要进行协作的,然而刚开始因为每一台计算机之间都是互相独立的。所以前面的人处理完了就需要将数据

  6. 【Elasticsearch基础】Elasticsearch索引、文档以及映射操作详解 - 2

    文章目录概念索引相关操作创建索引更新副本查看索引删除索引索引的打开与关闭收缩索引索引别名查询索引别名文档相关操作新建文档查询文档更新文档删除文档映射相关操作查询文档映射创建静态映射创建索引并添加映射概念es中有三个概念要清楚,分别为索引、映射和文档(不用死记硬背,大概有个印象就可以)索引可理解为MySQL数据库;映射可理解为MySQL的表结构;文档可理解为MySQL表中的每行数据静态映射和动态映射上面已经介绍了,映射可理解为MySQL的表结构,在MySQL中,向表中插入数据是需要先创建表结构的;但在es中不必这样,可以直接插入文档,es可以根据插入的文档(数据),动态的创建映射(表结构),这就

  7. c++基础-运算符 - 2

    目录1关系运算符2运算符优先级3关系表达式的书写代码实例:下面是面试中可能遇到的问题:1关系运算符C++中有6个关系运算符,用于比较两个值的大小关系,它们分别是:运算符描述==等于!=不等于小于>大于小于等于>=大于等于这些运算符返回一个布尔值,即true或false。例如,当x等于y时,x==y的结果为true,否则结果为false。2运算符优先级在C++中,关系运算符的优先级高于赋值运算符,但低于算术运算符。以下是关系运算符的优先级,从高到低排列:运算符描述>,,>=,关系运算符==,!=相等性运算符&&逻辑与`如果在表达式中有多个运算符,则按照优先级顺序依次进行运算。3关系表达式的书写在

  8. 计算机必读基础书籍 - 2

    一.计算机组成原理    这本书利用组合逻辑、同步时序逻辑电路设计的相关知识,从逻辑门开始逐步构建运算器、存储器、数据通路和控制器,最终集成为完整的CU原型系统,使读者从设计者的角度理解计算机部件构成及运行的基本原理,掌握软硬件协同的概念。    全书共9章,主要内容包括计算机系统概述、数据信息的表示、运算方法与运算器、存储系统、指令系统、中央处理器、指令流水线、总线系统、输入输出系统。1.计算机系统概述1.1计算机发展历程    计算机是一种能够按照事先存储的程序,自动、高速、准确地对相关信息进行处理的电子设备。1946年2月,世界上第一台电子数字计算机ENIAC(ElectronicNum

  9. 0基础学习软件测试有哪些建议 - 2

    其实现在基础的资料和视频到处都是,就是看你有没有认真的去找学习资源了,去哪里学习都是要看你个人靠谱不靠谱,再好的教程和老师,你自己学习不进去也是白搭在正式选择之前,大可以在各种学习网站里面找找学习资源先自己学习一下为什么选择学软件测试?同学们理由众多!大概分这几类:①不受开发语言、行业产品变化限制;②入门更简单,对零基础、女生都友好;③软件项目都需要测试人员,职业生涯稳;④学习周期短,但薪资并不低。要想“肩扛”一条线?需掌握三大技能:技能1:掌握测试流程,熟悉系统框架能提前与开发人员一起制定测试计划,通过测试左移,推动代码评审,代码审计,单元测试,自动化冒烟测试,来保证研发阶段的质量。技能2:

  10. ruby - Ruby基础知识 - 2

    Asitcurrentlystands,thisquestionisnotagoodfitforourQ&Aformat.Weexpectanswerstobesupportedbyfacts,references,orexpertise,butthisquestionwilllikelysolicitdebate,arguments,polling,orextendeddiscussion.Ifyoufeelthatthisquestioncanbeimprovedandpossiblyreopened,visitthehelpcenter提供指导。已关闭8年。什么是学习ruby语言

随机推荐