readmemh函数引用的txt格式_系统任务$readmemb和$readmemh

本文详细介绍了Verilog中的系统任务$readmemb和$readmemh,用于从txt文件中读取数据。这两个任务分别用于读取二进制和十六进制数据,要求文件格式特定。通过实例解释了如何使用它们初始化存储器,并讨论了不同格式的系统任务调用,以及如何处理数据文件中的地址和数据。同时提到了其他与文件操作相关的系统函数,如$fopen、$fclose、$fscanf等。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

系统函数$fopen用于打开一个文件,并还回一个整数指针.然后,$fdisplay就可以使用这个文件指针在文件中写入信息,写完后,则可以使用$fclose系统关闭这个文件

例如:

integer write_out_file;//定义一个文件指针

integer write_out_file=$fopen("write_out_file.txt");

$fdisplay(write_out_file,"@%h\n%h",addr,data);

$fclose("write_out_file");

以上语法是将addr,data分别显示在"@%h\n%h"中的2个%h的位置,并写入write_out_file文件指针所指向的write_out_file.txt中.

从文件中读取数据,可以用

$readmemb $readmemh 从文件中读入数据,该文件格式是一定的.

reg[7:0] data[47:0];

$readmemh("file_name.txt",data

);

就是将file_name.txt中的数据读入到data数组中,然后就可以使用这些数据了.

还有一种方式可以把指定的数据放入指定的存储器地址单元内,就是在存放数据的文本文件内,给相应的数据规定其内存地址,形式如下:

@address_in_hexadecimal   data

@2f             20

两个系统任务可以在仿真的任何时刻被执行使用,其使用格式共有以下六种:

1) $readmemb("",);

2) $readmemb("",,);

3) $readmemb("",,,);

4) $readmemh("",);

5) $readmemh("",,);

6) $readmemh("",,,);

在这两个系统任务中,被读取的数据文件的内容只能包含:空白位置(空格,换行,制表格(tab)和form-feeds),注释行(//形式的和形式的都允许),二进制或十六进制的数字。数字中不能包含位宽说明和格式说明,对于$readmemb系统任务,每个数字必须是二进制数字,对于$readmemh系统任务,每个数字必须是十六进制数字。数字中不定值x或X,高阻值z或Z,和下划线(_)的使用方法及代表的意义与一般Verilog

HDL程序中的用法及意义是一样的。另外数字必须用空白位置或注释行来分隔开。

在下面的讨论中,地址一词指对存贮器(memory)建模的数组的寻址指针。当数据文件被读取时,每一个被读取的数字都被存放到地址连续的存贮器单元中去。存贮器单元的存放地址范围由系统任务声明语句中的起始地址和结束地址来说明,每个数据的存放地址在数据文件中进行说明。当地址出现在数据文件中,其格式为字符“@”后跟上十六进制数。如:

@hh...h

对于这个十六进制的地址数中,允许大写和小写的数字。在字符“@”和数字之间不允许存在空白位置。可以在数据文件里出现多个地址。当系统任务遇到一个地址说明时,系统任务将该地址后的数据存放到存贮器中相应的地址单元中去。

对于上面六种系统任务格式,需补充说明以下五点:

1) 如果系统任务声明语句中和数据文件里都没有进行地址说明,则缺省的存放起始地址为该存贮器定义语句中的起始地址。数据文件里的数据被连续存放到该存贮器中,直到该存贮器单元存满为止或数据文件里的数据存完。

2) 如果系统任务中说明了存放的起始地址,没有说明存放的结束地址,则数据从起始地址开始存放,存放到该存贮器定义语句中的结束地址为止。

3) 如果在系统任务声明语句中,起始地址和结束地址都进行了说明,则数据文件里的数据按该起始地址开始存放到存贮器单元中,直到该结束地址,而不考虑该存贮器的定义语句中的起始地址和结束地址。

4) 如果地址信息在系统任务和数据文件里都进行了说明,那么数据文件里的地址必须在系统任务中地址参数声明的范围之内。否则将提示错误信息,并且装载数据到存贮器中的操作被中断。

5) 如果数据文件里的数据个数和系统任务中起始地址及结束地址暗示的数据个数不同的话,也要提示错误信息。

下面举例说明:

先定义一个有256个地址的字节存贮器 mem:

reg[7:0] mem[1:256];

下面给出的系统任务以各自不同的方式装载数据到存贮器mem中。

initial $readmemh("mem.data",mem);

initial $readmemh("mem.data",mem,16);

initial $readmemh("mem.data",mem,128,1);

第一条语句在仿真时刻为0时,将装载数据到以地址是1的存贮器单元为起始存放单元的存贮器中去。第二条语句将装载数据到以单元地址是16的存贮器单元为起始存放单元的存贮器中去,一直到地址是256的单元为止。第三条语句将从地址是128的单元开始装载数据,一直到地址为1的单元。在第三种情况中,当装载完毕,系统要检查在数据文件里是否有128个数据,如果没有,系统将提示错误信息。

"引用 参考"

1.打开文件

integer file_id;

file_id = fopen("file_path/file_name");

2.写入文件

//$fmonitor只要有变化就一直记录

$fmonitor(file_id, "%format_char", parameter);

eg:$fmonitor(file_id, "%m: %t in1=%d o1=%h", $time, in1, o1);

//$fwrite需要触发条件才记录

$fwrite(file_id, "%format_char", parameter);

//$fdisplay需要触发条件才记录

$fdisplay(file_id, "%format_char", parameter);

$fstrobe();

3.读取文件

integer file_id;

file_id = $fread("file_path/file_name", "r");

4.关闭文件

$fclose(fjile_id);

5.由文件设定存储器初值

$readmemh("file_name", memory_name"); //初始化数据为十六进制

$readmemb("file_name", memory_name"); //初始化数据为二进制

转:

提供了丰富的系统函数,这为Testbench的编写提供了方便。尤其是IEEE1364-2005,其系统级建模的能力更强。

以前我一般常用到的系统函数只有几个:$readmemb,$readmemh,$display,$fmonitor,$fwrite,$fopen,$fclose等。通常需要对文件作预处理,才能用于Testbench读取。今天又尝试了几个其他的文件输入输出函数,不需要对文件进行预处理,直接使用需要的文件,只对需要的部分进行读取。

$fseek,文件定位,可以从任意点对文件进行操作;

$fscanf,对文件一行进行读写。

下面是一些常见的应用:

1、读写文件

`timescale 1 ns/1 ns

module FileIO_tb;

integer fp_r, fp_w, cnt;

reg [7:0] reg1, reg2, reg3;

initial begin

fp_r = $fopen("data_in.txt", "r");

fp_w = $fopen("data_out.txt", "w");

while(!$feof(fp_r)) begin

cnt =

$fscanf(fp_r, "%d %d %d", reg1, reg2, reg3);

$display("%d

%d %d", reg1, reg2, reg3);

$fwrite(fp_w, "%d %d %d\n", reg3, reg2, reg1);

end

$fclose(fp_r);

$fclose(fp_w);

end

endmodule

2、

integer file, char;

reg eof;

initial begin

file =

$fopenr("myfile.txt");

eof = 0;

while (eof == 0) begin

char = $fgetc(file);

eof = $feof (file);

$display ("%s", char);

end

end

3、文件处理定位

`define SEEK_SET 0

`define SEEK_CUR 1

`define SEEK_END 2

integer file, offset, position, r;

r = $fseek(file, 0, `SEEK_SET);

r = $fseek(file, 0, `SEEK_CUR);

r = $fseek(file, 0, `SEEK_END);

r = $fseek(file, position, `SEEK_SET);

4、

integer r, file, start, count;

reg [15:0] mem[0:10], r16;

r = $fread(file, mem[0], start, count);

r = $fread(file, r16);

5、

integer file, position;

position = $ftell(file);

6、

integer file, r, a, b;

reg [80*8:1] string;

file = $fopenw("output.log");

r = $sformat(string, "Formatted %d %x", a, b);

r = $sprintf(string, "Formatted %d %x", a, b);

r = $fprintf(file, "Formatted %d %x", a, b);

7、

integer file, r;

file = $fopenw("output.log");

r = $fflush(file);

8、

// This is a pattern file - read_pattern.pat

// time bin dec hex

10: 001 1 1

20.0: 010 20 020

50.02: 111 5 FFF

62.345: 100 4 DEADBEEF

75.789: XXX 2 ZzZzZzZz

`timescale 1ns / 10 ps

`define EOF 32'hFFFF_FFFF

`define NULL 0

`define MAX_LINE_LENGTH 1000

module read_pattern;

integer file, c, r;

reg [3:0] bin;

reg [31:0] dec, hex;

real real_time;

reg [8*`MAX_LINE_LENGTH:0] line;

initial

begin :

file_block

$timeformat(-9, 3, "ns", 6);

$display("time bin decimal hex");

file =

$fopenr("read_pattern.pat");

if (file ==

`NULL) // If error opening file

disable file_block; // Just quit

c =

$fgetc(file);

while (c !=

`EOF)

begin

if (c == "/")

r = $fgets(line, `MAX_LINE_LENGTH, file);

else

begin

// Push the character back to the file then read the next time

r = $ungetc(c, file);

r = $fscanf(file," %f:\n", real_time);

// Wait until the absolute time in the file, then read stimulus

if ($realtime > real_time)

$display("Error - absolute time in file is out of order - %t",

real_time);

else

#(real_time - $realtime)

r = $fscanf(file," %b %d %h\n",bin,dec,hex);

end // if c else

c = $fgetc(file);

end // while not EOF

r =

$fcloser(file);

end //

initial

// Display changes to the signals

always @(bin or dec or hex)

$display("%t

%b %d %h", $realtime, bin, dec, hex);

endmodule // read_pattern

9、自动比较输出结果

`define EOF 32'hFFFF_FFFF

`define NULL 0

`define MAX_LINE_LENGTH 1000

module compare;

integer file, r;

reg a, b, expect, clock;

wire out;

reg [`MAX_LINE_LENGTH*8:1];

parameter cycle = 20;

initial

begin :

file_block

$display("Time Stim Expect Output");

clock =

0;

file =

$fopenr("compare.pat");

if (file ==

`NULL)

disable file_block;

r =

$fgets(line, MAX_LINE_LENGTH, file); // Skip comments

r =

$fgets(line, MAX_LINE_LENGTH, file);

while

(!$feof(file))

begin

// Wait until rising clock, read stimulus

@(posedge clock)

r = $fscanf(file, " %b %b %b\n", a, b, expect);

// Wait just before the end of cycle to do compare

#(cycle - 1)

$display("%d %b %b %b %b", $stime, a, b, expect, out);

$strobe_compare(expect, out);

end // while not EOF

r =

$fcloser(file);

$stop;

end //

initial

always #(cycle / 2) clock = !clock; // Clock generator

and #4 (out, a, b); // Circuit under test

endmodule // compare

10、从文件中读数据到mem(这个好像一般人用的最多了)

`define EOF 32'HFFFF_FFFF

`define MEM_SIZE 200_000

module load_mem;

integer file, i;

reg [7:0] mem[0:`MEM_SIZE];

reg [80*8:1] file_name;

initial

begin

file_name = "data.bin";

file = $fopenr(file_name);

i = $fread(file, mem[0]);

$display("Loaded entries \n", i);

i = $fcloser(file);

$stop;

end endmodule // load_mem

<think>好的,我现在需要帮助用户解答关于Verilog/VHDL中使用readmemh函数读取hex文件到存储器时是否支持.mem后缀名的问题。首先,我需要回忆一下Verilog中$readmemh$readmemb函数的用法,以及它们对文件后缀名的要求。 根据用户提供的引用[1]引用[2],$readmemh$readmemb函数在Verilog中是用来读取十六进制或二进制文件到存储器的。引用[1]提到函数的基本语法,其中文件路径需要正确。而引用[2]中的例子用了.txt后缀名,比如sin.txt,这说明实际使用时文件后缀名不一定是固定的。 接下来,我需要确认是否支持.mem后缀名。根据我所知道的Verilog标准,这两个函数并不关心文件的后缀名是什么,只要文件内容格式正确即可。比如,无论是.txt还是.mem,甚至是其他自定义后缀,只要文件内容符合十六进制或二进制的数据格式函数都能正确读取。因此,用户的问题答案是支持.mem后缀名。 然后,我需要检查用户提供的引用是否有相关说明。在引用[1][2]中,例子用了.txt,但并没有提到后缀名的限制。引用[4]中的Matlab代码生成的是sin.txt文件,同样没有限制。因此,可以确定后缀名不影响函数的使用,关键在内容格式。 接下来,关于VHDL方面,用户也询问了VHDL中的情况。VHDL没有内置的readmemh函数,通常需要使用TEXTIO库来读取文件。同样,文件后缀名在VHDL中也没有强制要求,主要取决于文件内容是否正确。例如,用户可以用.mem或.txt,只要数据格式正确即可。 然后,用户可能需要了解具体如何编写这些函数的使用方法,包括语法示例。比如,Verilog中使用$readmemh("file.mem", mem_array);,而VHDL则需要通过循环逐行读取,并转换数据到存储器数组。 最后,生成相关问题时要围绕文件后缀名、函数语法、数据格式转换以及不同语言(Verilog vs VHDL)的实现差异等方面。确保问题覆盖用户可能遇到的后续疑问,比如如何处理负数补码,如何生成hex文件等,这些在引用[4]中提到的Matlab方法可能有帮助。</think>在Verilog/VHDL中读取HEX文件到存储器的语法及文件后缀名要求如下: ### Verilog实现 1. **语法规则**: - `$readmemh`用于读取十六进制文件,文件内容格式要求: - 每行可以是十六进制数字(允许包含`0-9,a-f,A-F`以及`x`或`z`) - 注释以`//`或`/* */`标记 - 示例代码: ```verilog reg [7:0] mem [0:255]; // 定义8bit x 256的存储器 initial $readmemh("data.mem", mem); // 读取.mem文件[^2] ``` - **文件后缀名支持**: **支持任意后缀名**(如`.mem`、`.hex`、`.txt`),仅需路径正确且内容符合格式[^1] 2. **关键限制**: 存储深度需与数组定义匹配,否则会截断或填充默认值 ### VHDL实现 1. **语法规则**: VHDL无内置`readmemh`函数,需用`TEXTIO`库手动读取: ```vhdl process file f : text open read_mode is "data.mem"; variable line_data : line; begin for i in mem'range loop readline(f, line_data); hread(line_data, mem(i)); -- 十六进制读取[^4] end loop; file_close(f); end process; ``` 2. **文件后缀名支持**: 同样**无后缀限制**,但需文件内容为十六进制字符串 ### 文件格式对比 | 特性 | Verilog ($readmemh) | VHDL (TEXTIO) | |---------------|---------------------------|---------------------------| | 后缀名要求 | 任意(推荐.mem/.hex) | 任意(推荐.mem/.hex) | | 负数补码处理 | 需提前转换 | 需提前转换 | | 注释支持 | 支持`//`或`/* */`[^1] | 需手动过滤 | ### 示例文件内容(data.mem) ``` A3 // 地址0数据 1F /* 地址1数据 */ FF ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值