题解 | #非整数倍数据位宽转换24to128#

非整数倍数据位宽转换24to128

http://www.nowcoder.com/practice/6312169e30a645bba5d832c7313c64cc

题意整理

本题要求实现24bit数据至128bit数据的位宽转换电路,由接口电路图可知,valid_in信号会跟随指示data_in数据有效,同时要求输出valid_out信号跟随指示data_out数据。

观察时序图需要注意:

l  valid_out和data_out是在已存入5个数据且第6个数据到来后产生输出,且每个输入的数据中,低位数据会优先输出,图中第6个输入数据中的高8bit数据f2优先输出;

l  当数据缓存不满足128bit时,不会产生输出valid_out和data_out。

题解主体

要实现24bit数据至128bit数据的位宽转换,必须要用寄存器将先到达的数据进行缓存。24bit数据至128bit数据,相当于5个输入数据+第6个输入数据的拼接成一个输出数据,出于对资源的节省以及时序要求,采用120bit的寄存器(data_lock)进行数据缓存。

为什么采用120bit的寄存器进行数据缓存呢?

因为120bit刚好能存储5个24bit输入数据,当第6个数据到了之后,将寄存器内缓存的前5个数据数据和第6个输入数据的高8bit输出,剩下的低16bit在下个时钟周期存入到寄存器中。具体实现方式后面根据代码再解释。

为什么不采用更小bit位宽的寄存器存储而是120bit呢?原因如下:如果数据缓存寄存器的位宽小于128bit,那么寄存器的位宽就必须满足能存下5个输入数据,这样当第6个数据到来时,就直接将输入端的8bit的数据和已缓存的120bit数据,如果寄存器存不下5个输入数据,那么意味着将有1个以上的数据数据没有缓存,那么就会造成数据丢失。


根据时序推演,数据是在第6个、第11个、第16个数据到来之后输出,所以内部设计一个计数器(valid_cnt),用来指示数据接收状态。当检测到valid_in拉高时,valid_cnt加1,valid_cnt在0-15之间循环,valid_cnt复位值是0。当valid_cnt是5或10或15,且valid_in为高时,输出数据,valid_out拉高。

结合时序图,将电路转换成Verilog代码描述如下:


reg [119:0]          data_lock             ;

reg [3:0]       valid_cnt        ;


always @(posedge clk or negedge rst_n ) begin

       if(!rst_n)

              valid_cnt <= 'd0;

       else if(valid_in)begin

              if(valid_cnt == 4'd15)

                     valid_cnt <= 4'd0;

              else

                     valid_cnt <= valid_cnt + 1'd1;

       end

end


always @(posedge clk or negedge rst_n ) begin

       if(!rst_n) begin

              data_lock <= 'd0;

              data_out  <= 'd0;

       end

       else if(valid_in)begin

              if(valid_cnt == 4'd5)begin

                     data_lock <= {data_lock[119:16], data_in[15:0]};

                     data_out  <= {data_lock, data_in[23:16]};

              end

              if(valid_cnt == 4'd10)begin

                     data_lock <= {data_lock[119:8], data_in[7:0]};

                     data_out  <= {data_lock[111:0], data_in[23:8]};

              end

              if(valid_cnt == 4'd15)begin

                     data_lock <= {data_lock[119:24], data_in};

                     data_out  <= {data_lock[103:0], data_in};

              end

              else

                     data_lock <= {data_lock[95:0], data_in};

       end

end

always @(posedge clk or negedge rst_n ) begin

       if(!rst_n)

              valid_out <= 'd0;

       else if(valid_in && valid_cnt == 4'd5)

              valid_out <= 1'd1;

       else if(valid_in && valid_cnt == 4'd10)

              valid_out <= 1'd1;

       else if(valid_in && valid_cnt == 4'd15)

              valid_out <= 1'd1;

       else

              valid_out <= 'd0;

end


请注意以下代码:


else if(valid_in)begin

              if(valid_cnt == 4'd5)begin

                     data_lock <= {data_lock[119:16], data_in[15:0]};

                     data_out  <= {data_lock, data_in[23:16]};

              end

              if(valid_cnt == 4'd10)begin

                     data_lock <= {data_lock[119:8], data_in[7:0]};

                     data_out  <= {data_lock[111:0], data_in[23:8]};

              end

              if(valid_cnt == 4'd15)begin

                     data_lock <= {data_lock[119:24], data_in};

                     data_out  <= {data_lock[103:0], data_in};

              end

              else

                     data_lock <= {data_lock[95:0], data_in};

       end


此段代码是整个程序的核心代码。

if(valid_cnt == 4'd5),此部分代码,描述了第一个128bit的输出过程,以及将为输出的16bit输入数据缓存到寄存器中;

if(valid_cnt == 4'd10),此部分代码,描述了第二个128bit的输出过程,将{data_lock[111:0], data_in[23:8]}输出,且将未输出的8bit数据缓存进寄存器;

if(valid_cnt == 4'd15),描述了第三个128bit的输出过程,三个128bit数据正好是24*16,刚好16个输入数据拼成3个输出数据,为1个循环;

else       data_lock <= {data_lock[95:0], data_in};

此部分代码,描述了输入数据缓存进寄存器的过程。

参考答案

`timescale 1ns/1ns

module width_24to128(
	input 				clk 		,   
	input 				rst_n		,
	input				valid_in	,
	input	[23:0]		data_in		,
 
 	output	reg			valid_out	,
	output  reg [127:0]	data_out
);
reg 	[119:0]		data_lock		;
reg 	[3:0]		valid_cnt		;

always @(posedge clk or negedge rst_n ) begin
	if(!rst_n) 
		valid_cnt <= 'd0;
	else if(valid_in)begin
		if(valid_cnt == 4'd15)
			valid_cnt <= 4'd0;
		else
			valid_cnt <= valid_cnt + 1'd1;
	end 
end

always @(posedge clk or negedge rst_n ) begin
	if(!rst_n) begin 
		data_lock <= 'd0;
		data_out  <= 'd0;
	end 
	else if(valid_in)begin
		if(valid_cnt == 4'd5)begin
			data_lock <= {data_lock[119:16], data_in[15:0]};
			data_out  <= {data_lock, data_in[23:16]};
		end 
		if(valid_cnt == 4'd10)begin
			data_lock <= {data_lock[119:8], data_in[7:0]};
			data_out  <= {data_lock[111:0], data_in[23:8]};
		end 
		if(valid_cnt == 4'd15)begin
			data_lock <= {data_lock[119:24], data_in};
			data_out  <= {data_lock[103:0], data_in};
		end 
		else
			data_lock <= {data_lock[95:0], data_in};
	end 
end
always @(posedge clk or negedge rst_n ) begin
	if(!rst_n) 
		valid_out <= 'd0;
	else if(valid_in && valid_cnt == 4'd5)
		valid_out <= 1'd1;
	else if(valid_in && valid_cnt == 4'd10)
		valid_out <= 1'd1;
	else if(valid_in && valid_cnt == 4'd15)
		valid_out <= 1'd1;
	else
		valid_out <= 'd0;
end
endmodule


全部评论

相关推荐

昨天 11:02
已编辑
北方民族大学 全栈开发
“无名小卒,还是名扬天下?”我知道很多人都不觉得我能走到今天这一步,当然,也包括我自己。在我的人生里,有两部作品刻下了最深的烙印:《斗破苍穹》与《龙族》。它们总被人拿来对照:一边是萧炎的桀骜轻狂,一边是路明非的怯懦衰颓。有人说,天蚕土豆没见过魂天帝,但江南见过真凯撒。我时常觉得,自己就是那个衰小孩路明非。可路明非可以开挂,我不可以;我也无数次幻想过,能拥有萧炎那般年少轻狂的人生,可我没有他与生俱来的逆天天赋。我只是个平庸的普通人,一个看过《斗破苍穹》却开不了挂的路明非,只能一步一步往上爬。从我下定决心找实习的那一刻起,我就给自己定下了目标:“我一定要为字节跳动卖命.jpg”。萧炎有他的三年之约,我有我的两年半之约(其实是一年半)。2024.11.20,科大讯飞的第一封实习offer落进邮箱,我迈出了这场奔赴的第一步。2025.8.18,放弃百度转正的安稳机会,转身走进前路未卜的不确定里。我很感谢我在百度的mentor,是她从茫茫人海选中了我,给了我大厂实习的机会。即便有段时间我状态差、产出不理想,她依旧愿意认可我、希望我留下转正。2025.11.14,我选择走进字节跳动,以实习生的身份重新出发。2026.3.25&nbsp;-&nbsp;3.31,一周速通上海飞书,幸遇赏识我的伯乐,斩获Special&nbsp;Offer。被告知面试通过的那一刻,我的内心无比平静,就像这个offer本就该属于我。不是侥幸,是应得的。这一路,有人看轻过我的出身,不相信我能走到这里;也有人在我看不见前路的时候,替我举过灯。没有他们的鼓励与支撑,就没有今天站在这里的我。我看到了自强不息的激荡,那是一个双非的伟大乐章!我是雨夜迈巴赫,我要开启属于我的新篇章了。
在看牛客的本杰明很勇...:真心祝贺l总 我永远的偶像 我滴神
春招至今,你收到几个面试...
点赞 评论 收藏
分享
评论
30
2
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务