mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-31 03:32:29 +00:00 
			
		
		
		
	Refactor SF2 iobuf insertion, Add clkint insertion
Signed-off-by: Clifford Wolf <clifford@clifford.at>
This commit is contained in:
		
							parent
							
								
									b1b9edf5cc
								
							
						
					
					
						commit
						78762316aa
					
				
					 4 changed files with 153 additions and 84 deletions
				
			
		|  | @ -1,6 +1,6 @@ | |||
| #!/bin/bash | ||||
| set -ex | ||||
| yosys -p 'synth_sf2 -noclkbuf -top example -edif netlist.edn -vlog netlist.vm' example.v | ||||
| yosys -p 'synth_sf2 -top example -edif netlist.edn -vlog netlist.vm' example.v | ||||
| export LM_LICENSE_FILE=${LM_LICENSE_FILE:-1702@localhost} | ||||
| /opt/microsemi/Libero_SoC_v12.0/Libero/bin/libero SCRIPT:libero.tcl | ||||
| cp proj/designer/example/export/example.stp . | ||||
|  |  | |||
|  | @ -74,6 +74,13 @@ module CFG4 ( | |||
| 	assign Y = INIT >> {D, C, B, A}; | ||||
| endmodule | ||||
| 
 | ||||
| module CLKINT ( | ||||
| 	input A, | ||||
| 	output Y | ||||
| ); | ||||
| 	assign Y = A; | ||||
| endmodule | ||||
| 
 | ||||
| module CLKBUF ( | ||||
| 	input PAD, | ||||
| 	output Y | ||||
|  |  | |||
|  | @ -23,42 +23,8 @@ | |||
| USING_YOSYS_NAMESPACE | ||||
| PRIVATE_NAMESPACE_BEGIN | ||||
| 
 | ||||
| struct Sf2IobsPass : public Pass { | ||||
| 	Sf2IobsPass() : Pass("sf2_iobs", "SF2: insert IO buffers") { } | ||||
| 	void help() YS_OVERRIDE | ||||
| static void handle_iobufs(Module *module, bool clkbuf_mode) | ||||
| { | ||||
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | ||||
| 		log("\n"); | ||||
| 		log("    sf2_iobs [options] [selection]\n"); | ||||
| 		log("\n"); | ||||
| 		log("Add SF2 I/O buffers to top module IOs as needed.\n"); | ||||
| 		log("\n"); | ||||
| 		log("    -noclkbuf\n"); | ||||
| 		log("        Do not insert clock buffers\n"); | ||||
| 		log("\n"); | ||||
| 	} | ||||
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) YS_OVERRIDE | ||||
| 	{ | ||||
| 		bool noclkbuf_mode = false; | ||||
| 
 | ||||
| 		log_header(design, "Executing sf2_iobs pass (insert IO buffers).\n"); | ||||
| 
 | ||||
| 		size_t argidx; | ||||
| 		for (argidx = 1; argidx < args.size(); argidx++) | ||||
| 		{ | ||||
| 			if (args[argidx] == "-noclkbuf") { | ||||
| 				noclkbuf_mode = true; | ||||
| 				continue; | ||||
| 			} | ||||
| 			break; | ||||
| 		} | ||||
| 		extra_args(args, argidx, design); | ||||
| 
 | ||||
| 		Module *module = design->top_module(); | ||||
| 
 | ||||
| 		if (module == nullptr) | ||||
| 			log_cmd_error("No top module found.\n"); | ||||
| 
 | ||||
| 	SigMap sigmap(module); | ||||
| 
 | ||||
| 	pool<SigBit> clk_bits; | ||||
|  | @ -68,11 +34,13 @@ struct Sf2IobsPass : public Pass { | |||
| 
 | ||||
| 	for (auto cell : module->cells()) | ||||
| 	{ | ||||
| 			if (cell->type == "\\SLE") { | ||||
| 		if (clkbuf_mode && cell->type == "\\SLE") { | ||||
| 			for (auto bit : sigmap(cell->getPort("\\CLK"))) | ||||
| 				clk_bits.insert(bit); | ||||
| 		} | ||||
| 			if (cell->type.in("\\INBUF", "\\OUTBUF", "\\CLKBUF")) { | ||||
| 		if (cell->type.in("\\INBUF", "\\OUTBUF", "\\TRIBUF", "\\BIBUF", "\\CLKBUF", "\\CLKBIBUF", | ||||
| 				"\\INBUF_DIFF", "\\OUTBUF_DIFF", "\\BIBUFF_DIFF", "\\TRIBUF_DIFF", "\\CLKBUF_DIFF", | ||||
| 				"\\GCLKBUF", "\\GCLKBUF_DIFF", "\\GCLKBIBUF")) { | ||||
| 			for (auto bit : sigmap(cell->getPort("\\PAD"))) | ||||
| 				handled_io_bits.insert(bit); | ||||
| 		} | ||||
|  | @ -99,7 +67,7 @@ struct Sf2IobsPass : public Pass { | |||
| 			if (wire->port_output) { | ||||
| 				buf_type = "\\OUTBUF"; | ||||
| 				buf_port = "\\D"; | ||||
| 				} else if (clk_bits.count(canonical_bit) && !noclkbuf_mode) { | ||||
| 			} else if (clkbuf_mode && clk_bits.count(canonical_bit)) { | ||||
| 				buf_type = "\\CLKBUF"; | ||||
| 				buf_port = "\\Y"; | ||||
| 			} else { | ||||
|  | @ -130,6 +98,100 @@ struct Sf2IobsPass : public Pass { | |||
| 	for (auto &it : pad_bits) | ||||
| 		it.first->setPort("\\PAD", it.second); | ||||
| } | ||||
| 
 | ||||
| static void handle_clkint(Module *module) | ||||
| { | ||||
| 	SigMap sigmap(module); | ||||
| 
 | ||||
| 	pool<SigBit> clk_bits; | ||||
| 	vector<SigBit> handled_clk_bits; | ||||
| 
 | ||||
| 	for (auto cell : module->cells()) | ||||
| 	{ | ||||
| 		if (cell->type == "\\SLE") { | ||||
| 			for (auto bit : sigmap(cell->getPort("\\CLK"))) | ||||
| 				clk_bits.insert(bit); | ||||
| 		} | ||||
| 		if (cell->type.in("\\CLKBUF", "\\CLKBIBUF", "\\CLKBUF_DIFF", "\\GCLKBUF", "\\GCLKBUF_DIFF", "\\GCLKBIBUF", | ||||
| 				"\\CLKINT", "\\CLKINT_PRESERVE", "\\GCLKINT", "\\RCLKINT", "\\RGCLKINT")) { | ||||
| 			for (auto bit : sigmap(cell->getPort("\\Y"))) | ||||
| 				handled_clk_bits.push_back(bit); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	for (auto bit : handled_clk_bits) | ||||
| 		clk_bits.erase(bit); | ||||
| 
 | ||||
| 	for (auto cell : vector<Cell*>(module->cells())) | ||||
| 	for (auto &conn : cell->connections()) | ||||
| 	{ | ||||
| 		if (!cell->output(conn.first)) | ||||
| 			continue; | ||||
| 
 | ||||
| 		SigSpec sig = conn.second; | ||||
| 		bool did_something = false; | ||||
| 
 | ||||
| 		for (auto &bit : sig) { | ||||
| 			SigBit canonical_bit = sigmap(bit); | ||||
| 			if (clk_bits.count(canonical_bit)) { | ||||
| 				Cell *c = module->addCell(NEW_ID, "\\CLKINT"); | ||||
| 				SigBit new_bit = module->addWire(NEW_ID); | ||||
| 				c->setPort("\\A", new_bit); | ||||
| 				c->setPort("\\Y", bit); | ||||
| 				log("Added %s cell %s for clock signal %s.\n", log_id(c->type), log_id(c), log_signal(bit)); | ||||
| 				clk_bits.erase(canonical_bit); | ||||
| 				did_something = true; | ||||
| 				bit = new_bit; | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if (did_something) | ||||
| 			cell->setPort(conn.first, sig); | ||||
| 	} | ||||
| 
 | ||||
| 	for (auto bit : clk_bits) | ||||
| 		log_error("Failed to insert CLKINT for clock signal %s.\n", log_signal(bit)); | ||||
| } | ||||
| 
 | ||||
| struct Sf2IobsPass : public Pass { | ||||
| 	Sf2IobsPass() : Pass("sf2_iobs", "SF2: insert IO buffers") { } | ||||
| 	void help() YS_OVERRIDE | ||||
| 	{ | ||||
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | ||||
| 		log("\n"); | ||||
| 		log("    sf2_iobs [options] [selection]\n"); | ||||
| 		log("\n"); | ||||
| 		log("Add SF2 I/O buffers and global buffers to top module as needed.\n"); | ||||
| 		log("\n"); | ||||
| 		log("    -clkbuf\n"); | ||||
| 		log("        Insert PAD->global_net clock buffers\n"); | ||||
| 		log("\n"); | ||||
| 	} | ||||
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) YS_OVERRIDE | ||||
| 	{ | ||||
| 		bool clkbuf_mode = false; | ||||
| 
 | ||||
| 		log_header(design, "Executing sf2_iobs pass (insert IO buffers).\n"); | ||||
| 
 | ||||
| 		size_t argidx; | ||||
| 		for (argidx = 1; argidx < args.size(); argidx++) | ||||
| 		{ | ||||
| 			if (args[argidx] == "-clkbuf") { | ||||
| 				clkbuf_mode = true; | ||||
| 				continue; | ||||
| 			} | ||||
| 			break; | ||||
| 		} | ||||
| 		extra_args(args, argidx, design); | ||||
| 
 | ||||
| 		Module *module = design->top_module(); | ||||
| 
 | ||||
| 		if (module == nullptr) | ||||
| 			log_cmd_error("No top module found.\n"); | ||||
| 
 | ||||
| 		handle_iobufs(module, clkbuf_mode); | ||||
| 		handle_clkint(module); | ||||
| 	} | ||||
| } Sf2IobsPass; | ||||
| 
 | ||||
| PRIVATE_NAMESPACE_END | ||||
|  |  | |||
|  | @ -63,8 +63,8 @@ struct SynthSf2Pass : public ScriptPass | |||
| 		log("    -noiobs\n"); | ||||
| 		log("        run synthesis in \"block mode\", i.e. do not insert IO buffers\n"); | ||||
| 		log("\n"); | ||||
| 		log("    -noclkbuf\n"); | ||||
| 		log("        do not inser clock buffers, only simpe IO buffers\n"); | ||||
| 		log("    -clkbuf\n"); | ||||
| 		log("        insert direct PAD->global_net buffers\n"); | ||||
| 		log("\n"); | ||||
| 		log("    -retime\n"); | ||||
| 		log("        run 'abc' with -dff option\n"); | ||||
|  | @ -87,7 +87,7 @@ struct SynthSf2Pass : public ScriptPass | |||
| 		flatten = true; | ||||
| 		retime = false; | ||||
| 		iobs = true; | ||||
| 		clkbuf = true; | ||||
| 		clkbuf = false; | ||||
| 	} | ||||
| 
 | ||||
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) YS_OVERRIDE | ||||
|  | @ -134,8 +134,8 @@ struct SynthSf2Pass : public ScriptPass | |||
| 				iobs = false; | ||||
| 				continue; | ||||
| 			} | ||||
| 			if (args[argidx] == "-noclkbuf") { | ||||
| 				clkbuf = false; | ||||
| 			if (args[argidx] == "-clkbuf") { | ||||
| 				clkbuf = true; | ||||
| 				continue; | ||||
| 			} | ||||
| 			break; | ||||
|  | @ -210,9 +210,9 @@ struct SynthSf2Pass : public ScriptPass | |||
| 		if (check_label("map_iobs")) | ||||
| 		{ | ||||
| 			if (help_mode) | ||||
| 				run("sf2_iobs [-noclkbuf]", "(unless -noiobs)"); | ||||
| 				run("sf2_iobs [-clkbuf]", "(unless -noiobs)"); | ||||
| 			else if (iobs) | ||||
| 				run(clkbuf ? "sf2_iobs" : "sf2_iobs -noclkbuf"); | ||||
| 				run(clkbuf ? "sf2_iobs -clkbuf" : "sf2_iobs"); | ||||
| 			run("clean"); | ||||
| 		} | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue