mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-25 08:54:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			547 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			547 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  *  yosys -- Yosys Open SYnthesis Suite
 | |
|  *
 | |
|  *  Copyright (C) 2012  Claire Xenia Wolf <claire@yosyshq.com>
 | |
|  *
 | |
|  *  Permission to use, copy, modify, and/or distribute this software for any
 | |
|  *  purpose with or without fee is hereby granted, provided that the above
 | |
|  *  copyright notice and this permission notice appear in all copies.
 | |
|  *
 | |
|  *  THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 | |
|  *  WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 | |
|  *  MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 | |
|  *  ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | |
|  *  WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | |
|  *  ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 | |
|  *  OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include "kernel/register.h"
 | |
| #include "kernel/celltypes.h"
 | |
| #include "kernel/sigtools.h"
 | |
| #include "kernel/mem.h"
 | |
| #include "kernel/rtlil.h"
 | |
| #include "kernel/log.h"
 | |
| 
 | |
| #define MODE_ZERO     0
 | |
| #define MODE_ONE      1
 | |
| #define MODE_UNDEF    2
 | |
| #define MODE_RANDOM   3
 | |
| #define MODE_ANYSEQ   4
 | |
| #define MODE_ANYCONST 5
 | |
| 
 | |
| USING_YOSYS_NAMESPACE
 | |
| PRIVATE_NAMESPACE_BEGIN
 | |
| 
 | |
| static RTLIL::Wire * add_wire(RTLIL::Module *module, std::string name, int width, bool flag_input, bool flag_output)
 | |
| {
 | |
| 	RTLIL::Wire *wire = NULL;
 | |
| 	name = RTLIL::escape_id(name);
 | |
| 
 | |
| 	if (module->count_id(name) != 0)
 | |
| 	{
 | |
| 		log("Module %s already has such an object %s.\n", module->name, name);
 | |
| 		name += "$";
 | |
| 		return add_wire(module, name, width, flag_input, flag_output);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		wire = module->addWire(name, width);
 | |
| 		wire->port_input = flag_input;
 | |
| 		wire->port_output = flag_output;
 | |
| 
 | |
| 		if (flag_input || flag_output) {
 | |
| 			wire->port_id = module->wires_.size();
 | |
| 			module->fixup_ports();
 | |
| 		}
 | |
| 
 | |
| 		log("Added wire %s to module %s.\n", name, module->name);
 | |
| 	}
 | |
| 
 | |
| 	return wire;
 | |
| }
 | |
| 
 | |
| struct SetundefWorker
 | |
| {
 | |
| 	int next_bit_mode;
 | |
| 	uint32_t next_bit_state;
 | |
| 	vector<SigSpec*> siglist;
 | |
| 
 | |
| 	RTLIL::State next_bit()
 | |
| 	{
 | |
| 		if (next_bit_mode == MODE_ZERO)
 | |
| 			return RTLIL::State::S0;
 | |
| 
 | |
| 		if (next_bit_mode == MODE_ONE)
 | |
| 			return RTLIL::State::S1;
 | |
| 
 | |
| 		if (next_bit_mode == MODE_UNDEF)
 | |
| 			return RTLIL::State::Sx;
 | |
| 
 | |
| 		if (next_bit_mode == MODE_RANDOM)
 | |
| 		{
 | |
| 			// xorshift32
 | |
| 			next_bit_state ^= next_bit_state << 13;
 | |
| 			next_bit_state ^= next_bit_state >> 17;
 | |
| 			next_bit_state ^= next_bit_state << 5;
 | |
| 			log_assert(next_bit_state != 0);
 | |
| 
 | |
| 			return ((next_bit_state >> (next_bit_state & 15)) & 16) ? RTLIL::State::S0 : RTLIL::State::S1;
 | |
| 		}
 | |
| 
 | |
| 		log_abort();
 | |
| 	}
 | |
| 
 | |
| 	void operator()(RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		if (next_bit_mode == MODE_ANYSEQ || next_bit_mode == MODE_ANYCONST) {
 | |
| 			siglist.push_back(&sig);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire == NULL && bit.data > RTLIL::State::S1)
 | |
| 				bit = next_bit();
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct SetundefPass : public Pass {
 | |
| 	SetundefPass() : Pass("setundef", "replace undef values with defined constants") { }
 | |
| 	void help() override
 | |
| 	{
 | |
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | |
| 		log("\n");
 | |
| 		log("    setundef [options] [selection]\n");
 | |
| 		log("\n");
 | |
| 		log("This command replaces undef (x) constants with defined (0/1) constants.\n");
 | |
| 		log("\n");
 | |
| 		log("    -undriven\n");
 | |
| 		log("        also set undriven nets to constant values\n");
 | |
| 		log("\n");
 | |
| 		log("    -expose\n");
 | |
| 		log("        also expose undriven nets as inputs (use with -undriven)\n");
 | |
| 		log("\n");
 | |
| 		log("    -zero\n");
 | |
| 		log("        replace with bits cleared (0)\n");
 | |
| 		log("\n");
 | |
| 		log("    -one\n");
 | |
| 		log("        replace with bits set (1)\n");
 | |
| 		log("\n");
 | |
| 		log("    -undef\n");
 | |
| 		log("        replace with undef (x) bits, may be used with -undriven\n");
 | |
| 		log("\n");
 | |
| 		log("    -anyseq\n");
 | |
| 		log("        replace with $anyseq drivers (for formal)\n");
 | |
| 		log("\n");
 | |
| 		log("    -anyconst\n");
 | |
| 		log("        replace with $anyconst drivers (for formal)\n");
 | |
| 		log("\n");
 | |
| 		log("    -random <seed>\n");
 | |
| 		log("        replace with random bits using the specified integer as seed\n");
 | |
| 		log("        value for the random number generator.\n");
 | |
| 		log("\n");
 | |
| 		log("    -init\n");
 | |
| 		log("        also create/update init values for flip-flops\n");
 | |
| 		log("\n");
 | |
| 		log("    -params\n");
 | |
| 		log("        replace undef in cell parameters\n");
 | |
| 		log("\n");
 | |
| 	}
 | |
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) override
 | |
| 	{
 | |
| 		int got_value = 0;
 | |
| 		bool undriven_mode = false;
 | |
| 		bool expose_mode = false;
 | |
| 		bool init_mode = false;
 | |
| 		bool params_mode = false;
 | |
| 		SetundefWorker worker;
 | |
| 
 | |
| 		log_header(design, "Executing SETUNDEF pass (replace undef values with defined constants).\n");
 | |
| 
 | |
| 		size_t argidx;
 | |
| 		for (argidx = 1; argidx < args.size(); argidx++)
 | |
| 		{
 | |
| 			if (args[argidx] == "-undriven") {
 | |
| 				undriven_mode = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-expose") {
 | |
| 				expose_mode = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-zero") {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_ZERO;
 | |
| 				worker.next_bit_state = 0;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-one") {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_ONE;
 | |
| 				worker.next_bit_state = 0;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-anyseq") {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_ANYSEQ;
 | |
| 				worker.next_bit_state = 0;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-anyconst") {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_ANYCONST;
 | |
| 				worker.next_bit_state = 0;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-undef") {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_UNDEF;
 | |
| 				worker.next_bit_state = 0;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-init") {
 | |
| 				init_mode = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-params") {
 | |
| 				params_mode = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-random" && argidx+1 < args.size()) {
 | |
| 				got_value++;
 | |
| 				worker.next_bit_mode = MODE_RANDOM;
 | |
| 				worker.next_bit_state = atoi(args[++argidx].c_str()) + 1;
 | |
| 				for (int i = 0; i < 10; i++)
 | |
| 					worker.next_bit();
 | |
| 				continue;
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 		extra_args(args, argidx, design);
 | |
| 
 | |
| 		if (!got_value && expose_mode) {
 | |
| 			log("Using default as -undef with -expose.\n");
 | |
| 			got_value++;
 | |
| 			worker.next_bit_mode = MODE_UNDEF;
 | |
| 			worker.next_bit_state = 0;
 | |
| 		}
 | |
| 
 | |
| 		if (expose_mode && !undriven_mode)
 | |
| 			log_cmd_error("Option -expose must be used with option -undriven.\n");
 | |
| 		if (!got_value)
 | |
| 			log_cmd_error("One of the options -zero, -one, -anyseq, -anyconst, -random <seed>, or -expose must be specified.\n");
 | |
| 		else if (got_value > 1)
 | |
| 			log_cmd_error("Only one of the options -zero, -one, -anyseq, -anyconst, or -random <seed> can be specified.\n");
 | |
| 
 | |
| 		if (init_mode && (worker.next_bit_mode == MODE_ANYSEQ || worker.next_bit_mode == MODE_ANYCONST))
 | |
| 			log_cmd_error("The options -init and -anyseq / -anyconst are exclusive.\n");
 | |
| 
 | |
| 		for (auto module : design->selected_modules())
 | |
| 		{
 | |
| 			if (params_mode)
 | |
| 			{
 | |
| 				for (auto *cell : module->selected_cells()) {
 | |
| 					for (auto ¶meter : cell->parameters) {
 | |
| 						for (auto bit : parameter.second) {
 | |
| 							if (bit > RTLIL::State::S1)
 | |
| 								bit = worker.next_bit();
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (undriven_mode)
 | |
| 			{
 | |
| 				if (!module->processes.empty())
 | |
| 					log_error("The 'setundef' command can't operate in -undriven mode on modules with processes. Run 'proc' first.\n");
 | |
| 
 | |
| 				if (expose_mode)
 | |
| 				{
 | |
| 					SigMap sigmap(module);
 | |
| 					dict<SigBit, bool> wire_drivers;
 | |
| 					pool<SigBit> used_wires;
 | |
| 					SigPool undriven_signals;
 | |
| 
 | |
| 					for (auto cell : module->cells())
 | |
| 						for (auto &conn : cell->connections()) {
 | |
| 							SigSpec sig = sigmap(conn.second);
 | |
| 							if (cell->input(conn.first))
 | |
| 								for (auto bit : sig)
 | |
| 									if (bit.wire)
 | |
| 										used_wires.insert(bit);
 | |
| 							if (cell->output(conn.first))
 | |
| 								for (int i = 0; i < GetSize(sig); i++)
 | |
| 									if (sig[i].wire)
 | |
| 										wire_drivers[sig[i]] = true;
 | |
| 						}
 | |
| 
 | |
| 					for (auto wire : module->wires()) {
 | |
| 						if (wire->port_input) {
 | |
| 							SigSpec sig = sigmap(wire);
 | |
| 							for (int i = 0; i < GetSize(sig); i++)
 | |
| 								wire_drivers[sig[i]] = true;
 | |
| 						}
 | |
| 						if (wire->port_output) {
 | |
| 							SigSpec sig = sigmap(wire);
 | |
| 							for (auto bit : sig)
 | |
| 								if (bit.wire)
 | |
| 									used_wires.insert(bit);
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					pool<RTLIL::Wire*> undriven_wires;
 | |
| 					for (auto bit : used_wires)
 | |
| 						if (!wire_drivers.count(bit))
 | |
| 							undriven_wires.insert(bit.wire);
 | |
| 
 | |
| 					for (auto &it : undriven_wires)
 | |
| 						undriven_signals.add(sigmap(it));
 | |
| 
 | |
| 					for (auto &it : undriven_wires)
 | |
| 						if (it->port_input)
 | |
| 							undriven_signals.del(sigmap(it));
 | |
| 
 | |
| 					CellTypes ct(design);
 | |
| 					for (auto &it : module->cells_)
 | |
| 					for (auto &conn : it.second->connections())
 | |
| 						if (!ct.cell_known(it.second->type) || ct.cell_output(it.second->type, conn.first))
 | |
| 							undriven_signals.del(sigmap(conn.second));
 | |
| 
 | |
| 					RTLIL::SigSpec sig = undriven_signals.export_all();
 | |
| 					for (auto &c : sig.chunks()) {
 | |
| 						RTLIL::Wire * wire;
 | |
| 						if (c.wire->width == c.width) {
 | |
| 							wire = c.wire;
 | |
| 							wire->port_input = true;
 | |
| 						} else {
 | |
| 							string name = c.wire->name.str() + "$[" + std::to_string(c.width + c.offset) + ":" + std::to_string(c.offset) + "]";
 | |
| 							wire = add_wire(module, name, c.width, true, false);
 | |
| 							module->connect(RTLIL::SigSig(c, wire));
 | |
| 						}
 | |
| 						log("Exposing undriven wire %s as input.\n", wire->name);
 | |
| 					}
 | |
| 					module->fixup_ports();
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					SigMap sigmap(module);
 | |
| 					SigPool undriven_signals;
 | |
| 
 | |
| 					for (auto &it : module->wires_)
 | |
| 						undriven_signals.add(sigmap(it.second));
 | |
| 
 | |
| 					for (auto &it : module->wires_)
 | |
| 						if (it.second->port_input)
 | |
| 							undriven_signals.del(sigmap(it.second));
 | |
| 
 | |
| 					CellTypes ct(design);
 | |
| 					for (auto &it : module->cells_)
 | |
| 					for (auto &conn : it.second->connections())
 | |
| 						if (!ct.cell_known(it.second->type) || ct.cell_output(it.second->type, conn.first))
 | |
| 							undriven_signals.del(sigmap(conn.second));
 | |
| 
 | |
| 					RTLIL::SigSpec sig = undriven_signals.export_all();
 | |
| 					for (auto &c : sig.chunks()) {
 | |
| 						RTLIL::SigSpec bits;
 | |
| 						if (worker.next_bit_mode == MODE_ANYSEQ)
 | |
| 							bits = module->Anyseq(NEW_ID, c.width);
 | |
| 						else if (worker.next_bit_mode == MODE_ANYCONST)
 | |
| 							bits = module->Anyconst(NEW_ID, c.width);
 | |
| 						else
 | |
| 							for (int i = 0; i < c.width; i++)
 | |
| 								bits.append(worker.next_bit());
 | |
| 						module->connect(RTLIL::SigSig(c, bits));
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (init_mode)
 | |
| 			{
 | |
| 				SigMap sigmap(module);
 | |
| 				pool<SigBit> ffbits;
 | |
| 				pool<Wire*> initwires;
 | |
| 
 | |
| 				for (auto cell : module->cells())
 | |
| 				{
 | |
| 					if (!cell->is_builtin_ff())
 | |
| 						continue;
 | |
| 
 | |
| 					for (auto bit : sigmap(cell->getPort(ID::Q)))
 | |
| 						ffbits.insert(bit);
 | |
| 				}
 | |
| 
 | |
| 				auto process_initwires = [&]()
 | |
| 				{
 | |
| 					dict<Wire*, int> wire_weights;
 | |
| 
 | |
| 					for (auto wire : initwires)
 | |
| 					{
 | |
| 						int weight = 0;
 | |
| 
 | |
| 						for (auto bit : sigmap(wire))
 | |
| 							weight += ffbits.count(bit) ? +1 : -1;
 | |
| 
 | |
| 						wire_weights[wire] = weight;
 | |
| 					}
 | |
| 
 | |
| 					initwires.sort([&](Wire *a, Wire *b) { return wire_weights.at(a) > wire_weights.at(b); });
 | |
| 
 | |
| 					for (auto wire : initwires)
 | |
| 					{
 | |
| 						Const &initval = wire->attributes[ID::init];
 | |
| 						initval.resize(GetSize(wire), State::Sx);
 | |
| 
 | |
| 						for (int i = 0; i < GetSize(wire); i++) {
 | |
| 							SigBit bit = sigmap(SigBit(wire, i));
 | |
| 							if (initval[i] == State::Sx && ffbits.count(bit)) {
 | |
| 								initval.set(i, worker.next_bit());
 | |
| 								ffbits.erase(bit);
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						if (initval.is_fully_undef())
 | |
| 							wire->attributes.erase(ID::init);
 | |
| 					}
 | |
| 
 | |
| 					initwires.clear();
 | |
| 				};
 | |
| 
 | |
| 				for (int wire_types = 0; wire_types < 2; wire_types++)
 | |
| 				{
 | |
| 					// prioritize wires that already have an init attribute
 | |
| 					if (!ffbits.empty())
 | |
| 					{
 | |
| 						for (auto wire : module->wires())
 | |
| 						{
 | |
| 							if (wire->name[0] == (wire_types ? '\\' : '$'))
 | |
| 								continue;
 | |
| 
 | |
| 							if (!wire->attributes.count(ID::init))
 | |
| 								continue;
 | |
| 
 | |
| 							Const &initval = wire->attributes[ID::init];
 | |
| 							initval.resize(GetSize(wire), State::Sx);
 | |
| 
 | |
| 							if (initval.is_fully_undef()) {
 | |
| 								wire->attributes.erase(ID::init);
 | |
| 								continue;
 | |
| 							}
 | |
| 
 | |
| 							for (int i = 0; i < GetSize(wire); i++)
 | |
| 								if (initval[i] != State::Sx)
 | |
| 									ffbits.erase(sigmap(SigBit(wire, i)));
 | |
| 
 | |
| 							initwires.insert(wire);
 | |
| 						}
 | |
| 
 | |
| 						process_initwires();
 | |
| 					}
 | |
| 
 | |
| 					// next consider wires that completely contain bits to be initialized
 | |
| 					if (!ffbits.empty())
 | |
| 					{
 | |
| 						for (auto wire : module->wires())
 | |
| 						{
 | |
| 							if (wire->name[0] == (wire_types ? '\\' : '$'))
 | |
| 								continue;
 | |
| 
 | |
| 							for (auto bit : sigmap(wire))
 | |
| 								if (!ffbits.count(bit))
 | |
| 									goto next_wire;
 | |
| 
 | |
| 							initwires.insert(wire);
 | |
| 
 | |
| 						next_wire:
 | |
| 							continue;
 | |
| 						}
 | |
| 
 | |
| 						process_initwires();
 | |
| 					}
 | |
| 
 | |
| 					// finally use whatever wire we can find.
 | |
| 					if (!ffbits.empty())
 | |
| 					{
 | |
| 						for (auto wire : module->wires())
 | |
| 						{
 | |
| 							if (wire->name[0] == (wire_types ? '\\' : '$'))
 | |
| 								continue;
 | |
| 
 | |
| 							for (auto bit : sigmap(wire))
 | |
| 								if (ffbits.count(bit))
 | |
| 									initwires.insert(wire);
 | |
| 						}
 | |
| 
 | |
| 						process_initwires();
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				log_assert(ffbits.empty());
 | |
| 			}
 | |
| 
 | |
| 			if (worker.next_bit_mode == MODE_ANYSEQ || worker.next_bit_mode == MODE_ANYCONST)
 | |
| 			{
 | |
| 				// Do not add anyseq / anyconst to unused memory port clocks
 | |
| 				std::vector<Mem> memories = Mem::get_selected_memories(module);
 | |
| 				for (auto &mem : memories) {
 | |
| 					bool changed = false;
 | |
| 					for (auto &rd_port : mem.rd_ports) {
 | |
| 						if (!rd_port.clk_enable && rd_port.clk.is_fully_undef()) {
 | |
| 							changed = true;
 | |
| 							rd_port.clk = State::S0;
 | |
| 						}
 | |
| 					}
 | |
| 					for (auto &wr_port : mem.rd_ports) {
 | |
| 						if (!wr_port.clk_enable && wr_port.clk.is_fully_undef()) {
 | |
| 							changed = true;
 | |
| 							wr_port.clk = State::S0;
 | |
| 						}
 | |
| 					}
 | |
| 					if (changed)
 | |
| 						mem.emit();
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			for (auto &it : module->cells_)
 | |
| 				if (!it.second->get_bool_attribute(ID::xprop_decoder))
 | |
| 					it.second->rewrite_sigspecs(worker);
 | |
| 			for (auto &it : module->processes)
 | |
| 				it.second->rewrite_sigspecs(worker);
 | |
| 			for (auto &it : module->connections_) {
 | |
| 				worker(it.first);
 | |
| 				worker(it.second);
 | |
| 			}
 | |
| 
 | |
| 			if (worker.next_bit_mode == MODE_ANYSEQ || worker.next_bit_mode == MODE_ANYCONST)
 | |
| 			{
 | |
| 				vector<SigSpec*> siglist;
 | |
| 				siglist.swap(worker.siglist);
 | |
| 
 | |
| 				for (auto sigptr : siglist)
 | |
| 				{
 | |
| 					SigSpec &sig = *sigptr;
 | |
| 					int cursor = 0;
 | |
| 
 | |
| 					while (cursor < GetSize(sig))
 | |
| 					{
 | |
| 						int width = 0;
 | |
| 						while (cursor+width < GetSize(sig) && sig[cursor+width] == State::Sx)
 | |
| 							width++;
 | |
| 
 | |
| 						if (width > 0) {
 | |
| 							if (worker.next_bit_mode == MODE_ANYSEQ)
 | |
| 								sig.replace(cursor, module->Anyseq(NEW_ID, width));
 | |
| 							else
 | |
| 								sig.replace(cursor, module->Anyconst(NEW_ID, width));
 | |
| 							cursor += width;
 | |
| 						} else {
 | |
| 							cursor++;
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| } SetundefPass;
 | |
| 
 | |
| PRIVATE_NAMESPACE_END
 |