原创 通过文件读写方式实现Matlab和Modelsim的联合仿真

2009-11-16 22:10 8638 9 14 分类: FPGA/CPLD

????


    虽然Modelsim的功能非常强大,仿真的波形可以以多种形式进行显示,但是当涉及到数字信号处理的算法的仿真验证的时候,则显得有点不足。而进行数字信号处理是Matlab的强项,不但有大量的关于数字信号处理的函数,而且图形显示功能也很强大,所以在做数字信号处理算法的FPGA验证的时候借助Matlab会大大加快算法验证的速度。


    关于MatlabModelsim联合仿真,我从网上看到两种方法,一种是通过Link for Modelsim建立MatlabModelsim的联合仿真接口;另一种就是通过文件读写的方式实现MatlabModelsim的联合仿真。我没有仔细研究过第一种方法,我大概看了一下,感觉过程比较复杂,不过功能肯定也很强大,网上有一篇关于Link for Modelsim的文章https://static.assets-stash.eet-china.com/album/old-resources/2009/11/16/9e8d0364-20ed-4583-a85e-4d1fc50783a7.rar" target=_blank>,有兴趣的朋友可以去看一看。关于第二种方法,只是通过几个文件读写函数就可以实现了,而且基本可以满足当前仿真的要求,所以这里主要讨论一下我所使用的这种方法,希望能够抛砖引玉吧,因为我也只能算个初学者而已。


    1. Matlab产生数据用作Modelsim仿真


    在FPGA进行算法验证的时候,经常需要输入仿真数据,这些数据可以用FPGA产生,但是如果数据产生过程很复杂的话,需要耗费很大的精力,并且产生的数据的准确性也不能保证。例如,如果要验证一个通信接收机的相关算法,那么我们就需要先产生发送数据,也就是说得先做一个发射机,如果这个过程也由FPGA实现的话,也是一个很复杂的过程。这时候我们就可以借助Matlab,利用Matlab内部自带的各种函数,产生需要的信号,再经过定点化,就作为FPGA接收模块的输入信号了。这样做无疑会节约很多时间和精力。


    下面用一个简单的例子说明如何用Matlab产生的数据用作Modelsim仿真。


    首先利用matlab产生一个周期2568bit的正弦波数据,然后以16进制形式写入sin.txt文件


          N = 256;


          n = 1:256;


          x = fix(128 + (2^7 - 1) * sin(2*pi*n/N));


          fid = fopen('sin.txt','wt');


          fprintf(fid,'%x\n',x);


         fclose(fid);


    下图是截取的产生的数据文件的内容



    然后将产生的sin.txt文件复制到Modelsim的工程下,在Verilog文件中先定义一个8bit X 256数组,然后通过$readmemh命令,将文件中的数据读入,相关的Verilog代码如下:


reg [7:0] data_mem[0:255]; //定义一个8bit X 256的数组


initial


    begin


    $readmemh("sin.txt",data_mem); //sin.txt中的数据读入存储器data_mem


    end


    关于$readmemh的用法可以参见Verilog的参考书,这里就不详细说了。


后面就可以用data_mem作为你的测试数据了。例如可以通过以下代码,将data_mem的数据送给data_out


always @(posedge clk)


begin


    if(rst)


        begin


        data_out <= 8'd0;


        i <= 8'd0;


        end


    else


        begin


        data_out <= data_mem; //将存储器中的数据输出


        i <= i + 8'd1;


        end


end


    这样利用data_out就可以输出一个正弦波波形,下图是Molesim仿真产生的正弦波波形:


    2. MatlabModelsim仿真生成的数据进行分析


        MatlabModelsim仿真生成数据的处理也是通过文件读写实现的。即通过Verilog语句,将仿真过程中的某个信号写入文件,然后在Matlab中在把这个文件的数据读出来,就可以在Matlab中进行分析了。


    下图也通过一个简单的例子,说明一下整个过程。


    以下的Verilog语句实现将信号data_out的数据写入data_out.txt文件


    integer w_file;


    initial w_file = $fopen("data_out.txt");


    always @(i)


    begin


        $fdisplay(w_file,"%h",data_out);


        if(i == 8'd255)    //共写入256个数据


            $stop;


    end


    下图是截取的data_out.txt的部分内容:



    然后就可以编一小段Matlab的程序将data_out.txt中的数据读取进行分析了。下面一段Matlab的程序是将数据读取,并通过图形显示出数据的波形。


    fid = fopen('data_out.txt','r');


    for i = 1 : 256;


        num(i) = fscanf(fid, '%x', 1);    %这句话的意思是从fid所指的文件以16进制方式读出一个数据。


    end


    fclose(fid);


    plot(num);


    当利用fscanf函数时要注意两点


    第一:保证读取的数据格式和文件中保存的数据格式是相同的,例如这里文件中保存的格式是十六进制,所以读取的时候也应该以十六进制的形式读出。


    第二:要保证文件中数据的个数和设定的读取的数目(这里是256)保持一致。例如,要将生成文件data_out.txt中多余的换行符去掉(一般最后会多出一行),否则Matlab会将空的行也当做一个数据,从而两个数目不一致,导致Matlab报错。


    下图是Matlabdata_out.txt中的数据读出,并显示出的波形



    当然,有了Matlab这个强大的工具,也就可以很方便的看信号的频谱等信息了。


    另外在说一点,就是关于通过Verilog将数据写入文件有多种方法,上面用的是$fdisplay这个系统函数,当然还有$fmonitor$fwrite等几个命令,下面简单说一下这几个命令的不同。




  • $fdisplay

    这个命令需要有触发条件,才会把数据写入文件,例如,上例的触发条件就是always(i),当i变化的时候才写入。每写入一次数据会自动增加一个换行符。



  • $fmonitor

    这个命令不需要触发条件,只要有变化就可以将数据写入文件。例如可以通过以下语句:


    initial $fmonitor(w_file,"%h",data_out);


    这样可以将整个仿真过程产生的data_out数据都写入文件中。



  • $fwrite

    这个命令和$fdisplay基本相同,也是需要触发条件才会写入,不同的是每写入一个数据不会自动添加换行符。例如可以通过以下语句:


    always @(posedge clk)


    begin


        $fwrite(w_file,"%h\n",data_out);


    end


    关于这几个命令的详细介绍,大家可以参考Verilog的相关数据。


        简单总结一下上面用到的几个函数:



  • 关于Matlab的函数有:fopen fscanffclose
  • 关于Modelsim的函数有:$fopen $fclose$readmemh$readmemb$fmonitor$fdisplay$fwrite

    上面就是我关于MatlabModesim进行联合仿真的一些心得,如果大家还有其他更好的方法,希望不吝赐教啊!


    下面是相关的代码,包括Matlab的代码和Verilog的代码。


    

文章评论5条评论)

登录后参与讨论

用户1867243 2016-3-28 14:39

不错

用户377235 2012-10-19 21:27

我是新手 ,按照上面的方法但出现编译问题,楼主能不能给我个邮箱或者qq号 我想请教一些问题。

用户417383 2012-3-20 23:00

谢谢楼主,好厉害呀,高手!过程很详细,很清楚,就是可能是对verilog还是不太熟吧,在modelsim中编verilog时,编译总是有问题!希望可以向楼主请教!

用户1651241 2009-11-17 14:38

如果是在仿真环境中,就可以用我上面说的方法,将Modelsim仿真过程中生成的波形数据写入文件,然后用Matlab读出,就可以进行分析了。如果是想把通过SignalTap II抓到的波形用Matlab进行分析,也可以将SignalTap II找到的波形用Create SignalTap II list file命令保存下来,然后用Matlab把感兴趣的数据读出来,也可以进行分析,这在riple兄的一篇博文中提到过,你可以看一下这里http://blog.ednchina.com/riple/10833/category.aspx。谢谢你的提醒,我改天把如何把SignalTap II获取的数据用Matlab进行分析总结一下,再和大家分享。

coyoo 2009-11-17 10:55

不错啊! 请问有没有可能将仿真后的波形文件转换成matlab可识别的方式来进行分析呢? 我这个问题的实际意义是这样的,比如要做信号分析,具体比如FFT,进行时序仿真,或者通过SignalTapII抓到了输出波形,这时候一般需要分析频谱,虽然Altera以及Mentor的仿真工具支持模拟显示,但是毕竟没有matalb方便,是否可以通过matlab分析这些波形数据?谢谢 期待哦!!!!!
相关推荐阅读
用户1651241 2011-04-28 14:00
TMSI的作用
TMSI (Temporary Mobile Subscriber Identity),TMSI 是为了加强系统的保密性而在VLR 内分配的临时用户识别,它在某一VLR 区域内与IMSI 唯一对应,它...
用户1651241 2011-02-11 17:16
关于CDMA系统同步
终端首先应捕获附近基站的导频信道,这样就获得了基站经过路径delay的短码偏置和载波相位,有了这两个信息,终端就可以解调同步信道。 在同步信道传输的同步信息中,有三个和同步有关系的信息:LC_STAT...
用户1651241 2011-02-09 11:57
CDMA系统中的“掩码”和“扰码”的概念
在CDMA系统中,掩码用在长码发生器中,长码是利用一个42bit的m序列发生器与长码掩码进行模2加(即异或)来产生的。 掩码的作用就是使输出的m序列产生不同的偏移(通过用掩码也m序列模2加实现)。  ...
用户1651241 2011-02-08 21:42
Walsh码与M序列的区别
Walsh码互相关性好,不同Walsh码是完全正交的,即互相关为零。但是自相关性不好,虽然与自身的自相关为1,但是与自身延时后的码的互相关就不为0了。另外,Walsh码的数量较少,而M序列的数量较多,...
用户1651241 2011-01-30 23:06
CDMA功率控制技术
在功率控制技术,有以下几个术语: 前向功率控制 反向功率控制 开环功率控制 闭环功率控制 内环功率控制 外环功率控制 我们先以IS-95为例,用下面这个图来说明一下这几个概念之间的关系。 下面分别简单...
用户1651241 2011-01-30 22:55
转:CDMA中关于功率的几个概念
在CDMA中关于功率的这些概念比较多,容易混了,所以网上找了一些资料整理了一下。 由于下面的这个资料网上有很多,不知道原始的出处,所以就不一一注明了。 E是Energy(能量)的简称,c是...
我要评论
5
9
关闭 站长推荐上一条 /2 下一条