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(nosync) | ||||
| X(nowrshmsk) | ||||
| X(no_ram) | ||||
| X(no_rw_check) | ||||
| X(O) | ||||
| X(OFFSET) | ||||
|  |  | |||
|  | @ -148,6 +148,8 @@ void Mem::emit() { | |||
| 			for (int j = 0; j < (1 << wr_ports[i].wide_log2); j++) | ||||
| 				wr_port_xlat.push_back(i); | ||||
| 		for (auto &port : rd_ports) { | ||||
| 			for (auto attr: port.attributes) | ||||
| 				cell->attributes.insert(attr); | ||||
| 			if (port.cell) { | ||||
| 				module->remove(port.cell); | ||||
| 				port.cell = nullptr; | ||||
|  | @ -210,6 +212,8 @@ void Mem::emit() { | |||
| 		cell->setPort(ID::RD_ADDR, rd_addr); | ||||
| 		cell->setPort(ID::RD_DATA, rd_data); | ||||
| 		for (auto &port : wr_ports) { | ||||
| 			for (auto attr: port.attributes) | ||||
| 				cell->attributes.insert(attr); | ||||
| 			if (port.cell) { | ||||
| 				module->remove(port.cell); | ||||
| 				port.cell = nullptr; | ||||
|  | @ -246,6 +250,8 @@ void Mem::emit() { | |||
| 		cell->setPort(ID::WR_ADDR, wr_addr); | ||||
| 		cell->setPort(ID::WR_DATA, wr_data); | ||||
| 		for (auto &init : inits) { | ||||
| 			for (auto attr: init.attributes) | ||||
| 				cell->attributes.insert(attr); | ||||
| 			if (init.cell) { | ||||
| 				module->remove(init.cell); | ||||
| 				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.
 | ||||
| void MemMapping::determine_style() { | ||||
| 	kind = RamKind::Auto; | ||||
| 	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; | ||||
| 		log("found attribute 'lram' on memory %s.%s, forced mapping to huge RAM\n", log_id(mem.module->name), log_id(mem.memid)); | ||||
| 		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}) { | ||||
| 		if (mem.has_attribute(attr)) { | ||||
| 			Const val = mem.attributes.at(attr); | ||||
| 		find_attr = search_for_attribute(mem, attr); | ||||
| 		if (find_attr.first) { | ||||
| 			Const val = find_attr.second; | ||||
| 			if (val == 1) { | ||||
| 				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)); | ||||
|  | @ -526,8 +557,11 @@ void MemMapping::determine_style() { | |||
| 			return; | ||||
| 		} | ||||
| 	} | ||||
| 	if (mem.get_bool_attribute(ID::logic_block)) | ||||
| 	for (auto attr: {ID::logic_block, ID::no_ram}){ | ||||
| 		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.
 | ||||
|  |  | |||
|  | @ -1514,6 +1514,28 @@ end""" | |||
| 		{"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: | ||||
|     mf.write("ifneq ($(strip $(SEED)),)\n") | ||||
|     mf.write("SEEDOPT=-S$(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