mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-26 17:29:23 +00:00 
			
		
		
		
	memory_libmap: look for ram_style attributes on surrounding signals
This commit is contained in:
		
							parent
							
								
									d21c464ae4
								
							
						
					
					
						commit
						1b6d1e9419
					
				
					 5 changed files with 146 additions and 5 deletions
				
			
		|  | @ -140,6 +140,7 @@ X(nomem2reg) | ||||||
| X(nomeminit) | X(nomeminit) | ||||||
| X(nosync) | X(nosync) | ||||||
| X(nowrshmsk) | X(nowrshmsk) | ||||||
|  | X(no_ram) | ||||||
| X(no_rw_check) | X(no_rw_check) | ||||||
| X(O) | X(O) | ||||||
| X(OFFSET) | X(OFFSET) | ||||||
|  |  | ||||||
|  | @ -148,6 +148,8 @@ void Mem::emit() { | ||||||
| 			for (int j = 0; j < (1 << wr_ports[i].wide_log2); j++) | 			for (int j = 0; j < (1 << wr_ports[i].wide_log2); j++) | ||||||
| 				wr_port_xlat.push_back(i); | 				wr_port_xlat.push_back(i); | ||||||
| 		for (auto &port : rd_ports) { | 		for (auto &port : rd_ports) { | ||||||
|  | 			for (auto attr: port.attributes) | ||||||
|  | 				cell->attributes.insert(attr); | ||||||
| 			if (port.cell) { | 			if (port.cell) { | ||||||
| 				module->remove(port.cell); | 				module->remove(port.cell); | ||||||
| 				port.cell = nullptr; | 				port.cell = nullptr; | ||||||
|  | @ -210,6 +212,8 @@ void Mem::emit() { | ||||||
| 		cell->setPort(ID::RD_ADDR, rd_addr); | 		cell->setPort(ID::RD_ADDR, rd_addr); | ||||||
| 		cell->setPort(ID::RD_DATA, rd_data); | 		cell->setPort(ID::RD_DATA, rd_data); | ||||||
| 		for (auto &port : wr_ports) { | 		for (auto &port : wr_ports) { | ||||||
|  | 			for (auto attr: port.attributes) | ||||||
|  | 				cell->attributes.insert(attr); | ||||||
| 			if (port.cell) { | 			if (port.cell) { | ||||||
| 				module->remove(port.cell); | 				module->remove(port.cell); | ||||||
| 				port.cell = nullptr; | 				port.cell = nullptr; | ||||||
|  | @ -246,6 +250,8 @@ void Mem::emit() { | ||||||
| 		cell->setPort(ID::WR_ADDR, wr_addr); | 		cell->setPort(ID::WR_ADDR, wr_addr); | ||||||
| 		cell->setPort(ID::WR_DATA, wr_data); | 		cell->setPort(ID::WR_DATA, wr_data); | ||||||
| 		for (auto &init : inits) { | 		for (auto &init : inits) { | ||||||
|  | 			for (auto attr: init.attributes) | ||||||
|  | 				cell->attributes.insert(attr); | ||||||
| 			if (init.cell) { | 			if (init.cell) { | ||||||
| 				module->remove(init.cell); | 				module->remove(init.cell); | ||||||
| 				init.cell = nullptr; | 				init.cell = nullptr; | ||||||
|  |  | ||||||
|  | @ -481,18 +481,49 @@ void MemMapping::dump_config(MemConfig &cfg) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | std::pair<bool, Const> search_for_attribute(Mem mem, IdString attr) { | ||||||
|  | 	if (mem.has_attribute(attr)) | ||||||
|  | 		return std::make_pair(true, mem.attributes.at(attr)); | ||||||
|  | 	for (auto &port: mem.rd_ports){ | ||||||
|  | 		if (port.has_attribute(attr)) | ||||||
|  | 			return std::make_pair(true, port.attributes.at(attr)); | ||||||
|  | 		log_debug("looking for attribute %s on signal %s\n", log_id(attr), log_signal(port.data)); | ||||||
|  | 		for (SigBit bit: port.data) | ||||||
|  | 			if (bit.is_wire() && bit.wire->has_attribute(attr)) | ||||||
|  | 				return std::make_pair(true, bit.wire->attributes.at(attr)); | ||||||
|  | 		log_debug("looking for attribute %s on signal %s\n", log_id(attr), log_signal(port.data)); | ||||||
|  | 		for (SigBit bit: port.addr) | ||||||
|  | 			if (bit.is_wire() && bit.wire->has_attribute(attr)) | ||||||
|  | 				return std::make_pair(true, bit.wire->attributes.at(attr)); | ||||||
|  | 	} | ||||||
|  | 	for (auto &port: mem.wr_ports){ | ||||||
|  | 		if (port.has_attribute(attr)) | ||||||
|  | 			return std::make_pair(true, port.attributes.at(attr)); | ||||||
|  | 		log_debug("looking for attribute %s on signal %s\n", log_id(attr), log_signal(port.data)); | ||||||
|  | 		for (SigBit bit: port.data) | ||||||
|  | 			if (bit.is_wire() && bit.wire->has_attribute(attr)) | ||||||
|  | 				return std::make_pair(true, bit.wire->attributes.at(attr)); | ||||||
|  | 		for (SigBit bit: port.addr) | ||||||
|  | 			if (bit.is_wire() && bit.wire->has_attribute(attr)) | ||||||
|  | 				return std::make_pair(true, bit.wire->attributes.at(attr)); | ||||||
|  | 	} | ||||||
|  | 	return std::make_pair(false, Const()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Go through memory attributes to determine user-requested mapping style.
 | // Go through memory attributes to determine user-requested mapping style.
 | ||||||
| void MemMapping::determine_style() { | void MemMapping::determine_style() { | ||||||
| 	kind = RamKind::Auto; | 	kind = RamKind::Auto; | ||||||
| 	style = ""; | 	style = ""; | ||||||
| 	if (mem.get_bool_attribute(ID::lram)) { | 	auto find_attr = search_for_attribute(mem, ID::lram); | ||||||
|  | 	if (find_attr.first && find_attr.second.as_bool()) { | ||||||
| 		kind = RamKind::Huge; | 		kind = RamKind::Huge; | ||||||
| 		log("found attribute 'lram' on memory %s.%s, forced mapping to huge RAM\n", log_id(mem.module->name), log_id(mem.memid)); | 		log("found attribute 'lram' on memory %s.%s, forced mapping to huge RAM\n", log_id(mem.module->name), log_id(mem.memid)); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 	for (auto attr: {ID::ram_block, ID::rom_block, ID::ram_style, ID::rom_style, ID::ramstyle, ID::romstyle, ID::syn_ramstyle, ID::syn_romstyle}) { | 	for (auto attr: {ID::ram_block, ID::rom_block, ID::ram_style, ID::rom_style, ID::ramstyle, ID::romstyle, ID::syn_ramstyle, ID::syn_romstyle}) { | ||||||
| 		if (mem.has_attribute(attr)) { | 		find_attr = search_for_attribute(mem, attr); | ||||||
| 			Const val = mem.attributes.at(attr); | 		if (find_attr.first) { | ||||||
|  | 			Const val = find_attr.second; | ||||||
| 			if (val == 1) { | 			if (val == 1) { | ||||||
| 				kind = RamKind::NotLogic; | 				kind = RamKind::NotLogic; | ||||||
| 				log("found attribute '%s = 1' on memory %s.%s, disabled mapping to FF\n", log_id(attr), log_id(mem.module->name), log_id(mem.memid)); | 				log("found attribute '%s = 1' on memory %s.%s, disabled mapping to FF\n", log_id(attr), log_id(mem.module->name), log_id(mem.memid)); | ||||||
|  | @ -526,8 +557,11 @@ void MemMapping::determine_style() { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if (mem.get_bool_attribute(ID::logic_block)) | 	for (auto attr: {ID::logic_block, ID::no_ram}){ | ||||||
| 		kind = RamKind::Logic; | 		find_attr = search_for_attribute(mem, attr); | ||||||
|  | 		if (find_attr.first && find_attr.second.as_bool()) | ||||||
|  | 			kind = RamKind::Logic; | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Determine whether the memory can be mapped entirely to soft logic.
 | // Determine whether the memory can be mapped entirely to soft logic.
 | ||||||
|  |  | ||||||
|  | @ -1513,6 +1513,28 @@ end""" | ||||||
| 		["block_sp_full"], defs,  | 		["block_sp_full"], defs,  | ||||||
| 		{"RAM_BLOCK_SP": 1, "$*": add_logic} | 		{"RAM_BLOCK_SP": 1, "$*": add_logic} | ||||||
| 	)) | 	)) | ||||||
|  | 	 | ||||||
|  | ROM_CASE = """ | ||||||
|  | module rom(input clk, input [2:0] addr, {attr}output reg [7:0] data); | ||||||
|  | 
 | ||||||
|  | always @(posedge clk) begin | ||||||
|  | 	case (addr) | ||||||
|  | 		3'b000: data <= 8'h12; | ||||||
|  | 		3'b001: data <= 8'hAB; | ||||||
|  | 		3'b010: data <= 8'h42; | ||||||
|  | 		3'b011: data <= 8'h23; | ||||||
|  | 		3'b100: data <= 8'h66; | ||||||
|  | 		3'b101: data <= 8'hC0; | ||||||
|  | 		3'b110: data <= 8'h3F; | ||||||
|  | 		3'b111: data <= 8'h95; | ||||||
|  | 	endcase | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | endmodule | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | TESTS.append(Test("rom_case", ROM_CASE.format(attr=""), ["block_sdp"], [], {"RAM_BLOCK_SDP" : 0})) | ||||||
|  | TESTS.append(Test("rom_case_block", ROM_CASE.format(attr="(* rom_style = \"block\" *) "), ["block_sdp"], [], {"RAM_BLOCK_SDP" : 1})) | ||||||
| 
 | 
 | ||||||
| with open("run-test.mk", "w") as mf: | with open("run-test.mk", "w") as mf: | ||||||
|     mf.write("ifneq ($(strip $(SEED)),)\n") |     mf.write("ifneq ($(strip $(SEED)),)\n") | ||||||
|  |  | ||||||
							
								
								
									
										78
									
								
								tests/verific/rom_case.ys
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								tests/verific/rom_case.ys
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,78 @@ | ||||||
|  | verific -sv <<EOF | ||||||
|  | module rom(input clk, input [2:0] addr, (* ram_style = "block" *) output reg [7:0] data); | ||||||
|  | 
 | ||||||
|  | always @(posedge clk) begin | ||||||
|  | 	case (addr) | ||||||
|  | 		3'b000: data <= 8'h12; | ||||||
|  | 		3'b001: data <= 8'hAB; | ||||||
|  | 		3'b010: data <= 8'h42; | ||||||
|  | 		3'b011: data <= 8'h23; | ||||||
|  | 		3'b100: data <= 8'h66; | ||||||
|  | 		3'b101: data <= 8'hC0; | ||||||
|  | 		3'b110: data <= 8'h3F; | ||||||
|  | 		3'b111: data <= 8'h95; | ||||||
|  | 	endcase | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | endmodule | ||||||
|  | EOF | ||||||
|  | hierarchy -top rom | ||||||
|  | proc | ||||||
|  | opt | ||||||
|  | opt -full | ||||||
|  | memory -nomap | ||||||
|  | dump | ||||||
|  | memory_libmap -lib ../memlib/memlib_block_sdp.txt | ||||||
|  | memory_map | ||||||
|  | stat | ||||||
|  | select -assert-count 1 t:RAM_BLOCK_SDP | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | design -reset | ||||||
|  | 
 | ||||||
|  | verific -vhdl << | ||||||
|  | library IEEE; | ||||||
|  | use IEEE.std_logic_1164.all; | ||||||
|  | use ieee.std_logic_unsigned.all; | ||||||
|  | 
 | ||||||
|  | entity rom_example is | ||||||
|  | 	port ( | ||||||
|  | 		clk : in std_logic; | ||||||
|  | 		addr : in std_logic_vector(2 downto 0); | ||||||
|  | 		data : out std_logic_vector (7 downto 0) | ||||||
|  | 	); | ||||||
|  | end entity rom_example; | ||||||
|  | 
 | ||||||
|  | architecture rtl of rom_example is | ||||||
|  | 	attribute rom_style : string; | ||||||
|  | 	attribute rom_style of data : signal is "block"; | ||||||
|  | begin | ||||||
|  | 	 | ||||||
|  | 	p_rom : process(clk) | ||||||
|  | 	begin | ||||||
|  | 		if rising_edge(clk) then | ||||||
|  | 			case addr is | ||||||
|  | 				when "000" => data <= X"12"; | ||||||
|  | 				when "001" => data <= X"AB"; | ||||||
|  | 				when "010" => data <= X"42"; | ||||||
|  | 				when "011" => data <= X"23"; | ||||||
|  | 				when "100" => data <= X"66"; | ||||||
|  | 				when "101" => data <= X"C0"; | ||||||
|  | 				when "110" => data <= X"3F"; | ||||||
|  | 				when others => data <= X"95"; | ||||||
|  | 			end case; | ||||||
|  | 		end if; | ||||||
|  | 	end process p_rom; | ||||||
|  | 	 | ||||||
|  | end architecture rtl; | ||||||
|  | EOF | ||||||
|  | hierarchy -top rom | ||||||
|  | proc | ||||||
|  | opt | ||||||
|  | opt -full | ||||||
|  | memory -nomap | ||||||
|  | dump | ||||||
|  | memory_libmap -lib ../memlib/memlib_block_sdp.txt | ||||||
|  | memory_map | ||||||
|  | stat | ||||||
|  | select -assert-count 1 t:RAM_BLOCK_SDP | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue