OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 1984|回复: 6

[ALTERA] 请教一个资源分配问题,有附程序

[复制链接]

15

主题

79

帖子

0

精华

新手上路

积分
46
金钱
46
注册时间
2016-2-14
在线时间
36 小时
发表于 2023-7-5 18:31:34 | 显示全部楼层 |阅读模式
10金钱
/*通过除法计算值,资源占比非常大*/         
//                swp1_cnt1        <=         (1000/(sw_fre/10)*CLK_FREQ/1_000_000);
//                swp1_cnt2        <= (sw_dut>sw_dut_max_real) ? (1000/(sw_fre/10)*CLK_FREQ/1_000_000)*sw_dut_max_real/100 : (1000/(sw_fre/10)*CLK_FREQ/1_000_000)*sw_dut/100;               
/*通过直接赋值0,资源占比相对小*/               
                swp1_cnt1        <=        1'b0;
                swp1_cnt2        <=        1'b0;         

完整程序如下所示,请问是什么原因导致,或者怎么查看具体哪些程序占用了多少资源?


module sw_set(
    input              sys_clk,         
    input         sys_rst_n,         

   input                 sw_up_en,  
   input                 sw_en,
   input                 sw_p3_hl,  
   input                 sw_p4_hl,   
        input                                ad1_done_flag,                       
        input                                ad2_done_flag,                                  
       
        input        [15:0]        sw_fre,
        input        [15:0]        sw_dut,       
        input        [15:0]        sw_dt,       
        input        [15:0]        sw1_ctl,
        input        [15:0]        sw2_ctl,        
       
        input        [15:0]         ad1_data_m,
        input        [15:0]         ad2_data_m,
        input        [15:0]         ad1_set_rl,               
        input        [15:0]         ad2_set_rl,       

        output  reg   sw_p1,         
        output  reg   sw_p2,   
        output  reg   sw_p3,         
        output  reg   sw_p4,
        output  reg   sw_p5         

       
       
    );

`include "par.v"          
parameter  CLK_FREQ = 32_000_000;  

reg                ad1_done_flag_d0;
reg                ad1_done_flag_d1;       
wire        ad1_done_flag_flag;       

reg                sw_up_en_d0;
reg                sw_up_en_d1;       
wire        sw_up_en_flag;       

reg [35:0]         swp1_cnt;        //计时
reg [35:0]         swp1_cnt1;        //周期
reg [35:0]         swp1_cnt2;        //开启时间
reg [35:0]         swp1_cnt3;        //死区时间

reg [35:0]         swp3_cnt1;        //开启时间
reg [35:0]         swp4_cnt1;        //开启时间




reg [15:0]        sw_dut_max_real;               

reg                        sw_up_en1;
reg                sw_up_en1_d0;
reg                sw_up_en1_d1;       
wire        sw_up_en1_flag;       

assign ad1_done_flag_flag = (~ad1_done_flag_d1) & ad1_done_flag_d0;

always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n)
        begin
        ad1_done_flag_d0 <= 1'b0;                                 
        ad1_done_flag_d1 <= 1'b0;
    end                                                      
    else
        begin                                               
        ad1_done_flag_d0 <= ad1_done_flag;                              
        ad1_done_flag_d1 <= ad1_done_flag_d0;                           
    end
end       

assign sw_up_en_flag = (~sw_up_en_d1) & sw_up_en_d0;

always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n)
        begin
        sw_up_en_d0 <= 1'b0;                                 
        sw_up_en_d1 <= 1'b0;
    end                                                      
    else
        begin                                               
        sw_up_en_d0 <= sw_up_en;                              
        sw_up_en_d1 <= sw_up_en_d0;                           
    end
end       
               
       
       
       
       
       
//数据更新,更新占空比和死去时间的算法       
always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n)
         begin                             
                sw_dut_max_real<=1'b0;
                swp1_cnt3<=CLK_FREQ/1_000_000*sw_dt/10;
    end  
         else if(sw_up_en_flag)
         begin        

                        if((1000/(sw_fre/10)*CLK_FREQ/1_000_000) > (CLK_FREQ/1_000_000*sw_dt/10*2))
                        begin
//                        sw_dut_max_real        <= (((1000/sw_fre*10*CLK_FREQ/1_000_000)-(CLK_FREQ/1_000_000*sw_dt/10*2))/(1000/sw_fre*10*CLK_FREQ/1_000_000)*100>`sw_dut_max)? `sw_dut_max:sw_dut_max_real;
                                if(((1000/sw_fre*10*CLK_FREQ/1_000_000)-(CLK_FREQ/1_000_000*sw_dt/10*2))/(1000/sw_fre*10*CLK_FREQ/1_000_000)*100>`sw_dut_max)
                                begin
                                sw_dut_max_real        <=        `sw_dut_max;
                                swp1_cnt3                        <=        CLK_FREQ/1_000_000*sw_dt/10;       
                                sw_up_en1                        <=        1'b1;        
                                end
                                else
                                begin
                                sw_dut_max_real        <=        sw_dut;
                                swp1_cnt3                        <=        CLK_FREQ/1_000_000*sw_dt/10;       
                                sw_up_en1                        <=        1'b1;                                
                                end
                        end
                        else
                        begin
                        sw_up_en1<=1'b0;
                        swp1_cnt3<=swp1_cnt3;
                        end                       

         end  
         else         
         begin
                sw_up_en1        <=        1'b0;       
                swp1_cnt3        <=swp1_cnt3;       
         end         
end

       
assign sw_up_en1_flag = (~sw_up_en1_d1) & sw_up_en1_d0;

always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n)
        begin
        sw_up_en1_d0 <= 1'b0;                                 
        sw_up_en1_d1 <= 1'b0;
    end                                                      
    else
        begin                                               
        sw_up_en1_d0 <= sw_up_en1;                              
        sw_up_en1_d1 <= sw_up_en1_d0;                           
    end
end               
       
//根据占空比和死区时间计算计数值
always @(posedge sys_clk or negedge sys_rst_n) begin         
    if (!sys_rst_n)
         begin                             
                swp1_cnt1        <=        1'b0;
                swp1_cnt2        <=        1'b0;       
    end  
         else if(sw_up_en1_flag)
         begin  
/*通过除法计算值,资源占比非常大*/         
//                swp1_cnt1        <=         (1000/(sw_fre/10)*CLK_FREQ/1_000_000);
//                swp1_cnt2        <= (sw_dut>sw_dut_max_real) ? (1000/(sw_fre/10)*CLK_FREQ/1_000_000)*sw_dut_max_real/100 : (1000/(sw_fre/10)*CLK_FREQ/1_000_000)*sw_dut/100;               
/*通过直接赋值0,资源占比相对小*/               
                swp1_cnt1        <=        1'b0;
                swp1_cnt2        <=        1'b0;                        
         end  
         else         
         begin  
                swp1_cnt1        <=        swp1_cnt1;
                swp1_cnt2        <=        swp1_cnt2;                  
         end         
end       



always @(posedge sys_clk or negedge sys_rst_n)
begin
    if (!sys_rst_n)
        swp1_cnt         <= 32'd0;          
         else if(swp1_cnt<swp1_cnt1)
                        swp1_cnt<= swp1_cnt+32'd1;
         else
                        swp1_cnt<= 32'd0;         
end


       
//pw1 pw2输出
always @(posedge sys_clk or negedge sys_rst_n)
begin
    if (!sys_rst_n)
         begin
                  sw_p1                 <= 1'b1;          
    end
         else if(sw_en==1'b1)
         begin
                if(sw1_ctl        ==        8'd0)
                begin
                        if (swp1_cnt < swp1_cnt2)
                                sw_p1         <= 1'b0;                
                        else
                                sw_p1         <= 1'b1;                 
                end
                else if(sw1_ctl        ==        8'd1)
                        sw_p1         <= 1'b0;          
                else if(sw1_ctl        ==        8'd2)
                        sw_p1         <= 1'b1;          
                else
                        sw_p1         <= sw_p1;
        end
        else
        sw_p1         <= 1'b1;        
         
       
end


cnt1 2除法计算

cnt1 2除法计算
2.png

最佳答案

查看完整内容[请看2#楼]

你那个计算的挺复杂,乘除法是比较耗资源的。你可以看看那个计算能不能再化简一下,毕竟每个计算都是靠逻辑资源来实现的
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

3

主题

2016

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5622
金钱
5622
注册时间
2018-10-21
在线时间
1592 小时
发表于 2023-7-5 18:31:35 | 显示全部楼层
你那个计算的挺复杂,乘除法是比较耗资源的。你可以看看那个计算能不能再化简一下,毕竟每个计算都是靠逻辑资源来实现的
回复

使用道具 举报

15

主题

79

帖子

0

精华

新手上路

积分
46
金钱
46
注册时间
2016-2-14
在线时间
36 小时
 楼主| 发表于 2023-7-6 17:01:12 | 显示全部楼层
QinQZ 发表于 2023-7-5 18:31
你那个计算的挺复杂,乘除法是比较耗资源的。你可以看看那个计算能不能再化简一下,毕竟每个计算都是靠逻辑 ...

请问,您平时用FPGA做乘除法的时候,是怎么实现呢?还要外挂一个芯片吗?
另外,如果在一个always块里边:
几条赋值  a<= 1'b1;
               b<= 1'b1;  
               c<= 1'b1;
然后紧接一个判断句 if(d>1'b1)

这样会有问题吗?有什么风险吗?
回复

使用道具 举报

3

主题

2016

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5622
金钱
5622
注册时间
2018-10-21
在线时间
1592 小时
发表于 2023-7-6 17:41:20 | 显示全部楼层
why3354503 发表于 2023-7-6 17:01
请问,您平时用FPGA做乘除法的时候,是怎么实现呢?还要外挂一个芯片吗?
另外,如果在一个always块里边 ...

乘除法尽量搞简单点,能化简就化简,还有就是尽量用移位代替除法,代替不了也可以尝试用软件带的乘除法IP核来实现
回复

使用道具 举报

3

主题

2016

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5622
金钱
5622
注册时间
2018-10-21
在线时间
1592 小时
发表于 2023-7-6 17:47:22 | 显示全部楼层
why3354503 发表于 2023-7-6 17:01
请问,您平时用FPGA做乘除法的时候,是怎么实现呢?还要外挂一个芯片吗?
另外,如果在一个always块里边 ...

没啥风险啊,这不是熟常规操作嘛
回复

使用道具 举报

15

主题

79

帖子

0

精华

新手上路

积分
46
金钱
46
注册时间
2016-2-14
在线时间
36 小时
 楼主| 发表于 2023-7-19 22:31:52 | 显示全部楼层
QinQZ 发表于 2023-7-6 17:47
没啥风险啊,这不是熟常规操作嘛

FPGA不是并行处理吗?那么ABC的赋值和if判断句是同时进行的?还是先同时进行ABC赋值,再进行if判断句?
回复

使用道具 举报

3

主题

2016

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
5622
金钱
5622
注册时间
2018-10-21
在线时间
1592 小时
发表于 2023-7-20 13:22:06 | 显示全部楼层
why3354503 发表于 2023-7-19 22:31
FPGA不是并行处理吗?那么ABC的赋值和if判断句是同时进行的?还是先同时进行ABC赋值,再进行if判断句?

都是同时执行的
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2024-11-25 23:48

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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