mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-31 19:52:31 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			672 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			672 lines
		
	
	
	
		
			18 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 "blifparse.h"
 | |
| 
 | |
| YOSYS_NAMESPACE_BEGIN
 | |
| 
 | |
| const int lut_input_plane_limit = 12;
 | |
| 
 | |
| static bool read_next_line(char *&buffer, size_t &buffer_size, int &line_count, std::istream &f)
 | |
| {
 | |
| 	string strbuf;
 | |
| 	int buffer_len = 0;
 | |
| 	buffer[0] = 0;
 | |
| 
 | |
| 	while (1)
 | |
| 	{
 | |
| 		buffer_len += strlen(buffer + buffer_len);
 | |
| 		while (buffer_len > 0 && (buffer[buffer_len-1] == ' ' || buffer[buffer_len-1] == '\t' ||
 | |
| 				buffer[buffer_len-1] == '\r' || buffer[buffer_len-1] == '\n'))
 | |
| 			buffer[--buffer_len] = 0;
 | |
| 
 | |
| 		if (buffer_size-buffer_len < 4096) {
 | |
| 			buffer_size *= 2;
 | |
| 			buffer = (char*)realloc(buffer, buffer_size);
 | |
| 		}
 | |
| 
 | |
| 		if (buffer_len == 0 || buffer[buffer_len-1] == '\\') {
 | |
| 			if (buffer_len > 0 && buffer[buffer_len-1] == '\\')
 | |
| 				buffer[--buffer_len] = 0;
 | |
| 			line_count++;
 | |
| 			if (!std::getline(f, strbuf))
 | |
| 				return false;
 | |
| 			while (buffer_size-buffer_len < strbuf.size()+1) {
 | |
| 				buffer_size *= 2;
 | |
| 				buffer = (char*)realloc(buffer, buffer_size);
 | |
| 			}
 | |
| 			strcpy(buffer+buffer_len, strbuf.c_str());
 | |
| 		} else
 | |
| 			return true;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static std::pair<RTLIL::IdString, int> wideports_split(std::string name)
 | |
| {
 | |
| 	int pos = -1;
 | |
| 
 | |
| 	if (name.empty() || name.back() != ']')
 | |
| 		goto failed;
 | |
| 
 | |
| 	for (int i = 0; i+1 < GetSize(name); i++) {
 | |
| 		if (name[i] == '[')
 | |
| 			pos = i;
 | |
| 		else if (name[i] != '-' && (name[i] < '0' || name[i] > '9'))
 | |
| 			pos = -1;
 | |
| 		else if (name[i] == '-' && ((i != pos+1) || name[i+1] == ']'))
 | |
| 			pos = -1;
 | |
| 		else if (i == pos+2 && name[i] == '0' && name[i-1] == '-')
 | |
| 			pos = -1;
 | |
| 		else if (i == pos+1 && name[i] == '0' && name[i+1] != ']')
 | |
| 			pos = -1;
 | |
| 	}
 | |
| 
 | |
| 	if (pos >= 0)
 | |
| 		return std::pair<RTLIL::IdString, int>("\\" + name.substr(0, pos), atoi(name.c_str() + pos+1));
 | |
| 
 | |
| failed:
 | |
| 	return std::pair<RTLIL::IdString, int>(RTLIL::IdString(), 0);
 | |
| }
 | |
| 
 | |
| void parse_blif(RTLIL::Design *design, std::istream &f, IdString dff_name, bool run_clean, bool sop_mode, bool wideports)
 | |
| {
 | |
| 	RTLIL::Module *module = nullptr;
 | |
| 	RTLIL::Const *lutptr = NULL;
 | |
| 	RTLIL::Cell *sopcell = NULL;
 | |
| 	RTLIL::Cell *lastcell = nullptr;
 | |
| 	RTLIL::State lut_default_state = RTLIL::State::Sx;
 | |
| 	std::string err_reason;
 | |
| 	int blif_maxnum = 0, sopmode = -1;
 | |
| 
 | |
| 	auto blif_wire = [&](const std::string &wire_name) -> Wire*
 | |
| 	{
 | |
| 		if (wire_name[0] == '$')
 | |
| 		{
 | |
| 			for (int i = 0; i+1 < GetSize(wire_name); i++)
 | |
| 			{
 | |
| 				if (wire_name[i] != '$')
 | |
| 					continue;
 | |
| 
 | |
| 				int len = 0;
 | |
| 				while (i+len+1 < GetSize(wire_name) && '0' <= wire_name[i+len+1] && wire_name[i+len+1] <= '9')
 | |
| 					len++;
 | |
| 
 | |
| 				if (len > 0) {
 | |
| 					string num_str = wire_name.substr(i+1, len);
 | |
| 					int num = atoi(num_str.c_str()) & 0x0fffffff;
 | |
| 					blif_maxnum = std::max(blif_maxnum, num);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		IdString wire_id = RTLIL::escape_id(wire_name);
 | |
| 		Wire *wire = module->wire(wire_id);
 | |
| 
 | |
| 		if (wire == nullptr)
 | |
| 			wire = module->addWire(wire_id);
 | |
| 
 | |
| 		return wire;
 | |
| 	};
 | |
| 
 | |
| 	dict<RTLIL::IdString, RTLIL::Const> *obj_attributes = nullptr;
 | |
| 	dict<RTLIL::IdString, RTLIL::Const> *obj_parameters = nullptr;
 | |
| 
 | |
| 	dict<RTLIL::IdString, std::pair<int, bool>> wideports_cache;
 | |
| 
 | |
| 	size_t buffer_size = 4096;
 | |
| 	char *buffer = (char*)malloc(buffer_size);
 | |
| 	int line_count = 0;
 | |
| 
 | |
| 	while (1)
 | |
| 	{
 | |
| 		if (!read_next_line(buffer, buffer_size, line_count, f)) {
 | |
| 			if (module != nullptr)
 | |
| 				goto error;
 | |
| 			free(buffer);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 	continue_without_read:
 | |
| 		if (buffer[0] == '#')
 | |
| 			continue;
 | |
| 
 | |
| 		if (buffer[0] == '.')
 | |
| 		{
 | |
| 			if (lutptr) {
 | |
| 				for (auto bit : *lutptr)
 | |
| 					if (bit == RTLIL::State::Sx)
 | |
| 						bit = lut_default_state;
 | |
| 				lutptr = NULL;
 | |
| 				lut_default_state = RTLIL::State::Sx;
 | |
| 			}
 | |
| 
 | |
| 			if (sopcell) {
 | |
| 				sopcell = NULL;
 | |
| 				sopmode = -1;
 | |
| 			}
 | |
| 
 | |
| 			char *cmd = strtok(buffer, " \t\r\n");
 | |
| 
 | |
| 			if (!strcmp(cmd, ".model")) {
 | |
| 				if (module != nullptr)
 | |
| 					goto error;
 | |
| 				module = new RTLIL::Module;
 | |
| 				lastcell = nullptr;
 | |
| 				char *name = strtok(NULL, " \t\r\n");
 | |
| 				if (name == nullptr)
 | |
| 					goto error;
 | |
| 				module->name = RTLIL::escape_id(name);
 | |
| 				obj_attributes = &module->attributes;
 | |
| 				obj_parameters = nullptr;
 | |
| 				if (design->module(module->name))
 | |
| 					log_error("Duplicate definition of module %s in line %d!\n", log_id(module->name), line_count);
 | |
| 				design->add(module);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (module == nullptr)
 | |
| 				goto error;
 | |
| 
 | |
| 			if (!strcmp(cmd, ".blackbox"))
 | |
| 			{
 | |
| 				module->attributes[ID::blackbox] = RTLIL::Const(1);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".end"))
 | |
| 			{
 | |
| 				for (auto &wp : wideports_cache)
 | |
| 				{
 | |
| 					auto name = wp.first;
 | |
| 					int width = wp.second.first;
 | |
| 					bool isinput = wp.second.second;
 | |
| 
 | |
| 					RTLIL::Wire *wire = module->addWire(name, width);
 | |
| 					wire->port_input = isinput;
 | |
| 					wire->port_output = !isinput;
 | |
| 
 | |
| 					for (int i = 0; i < width; i++) {
 | |
| 						RTLIL::IdString other_name = name.str() + stringf("[%d]", i);
 | |
| 						RTLIL::Wire *other_wire = module->wire(other_name);
 | |
| 						if (other_wire) {
 | |
| 							other_wire->port_input = false;
 | |
| 							other_wire->port_output = false;
 | |
| 							if (isinput)
 | |
| 								module->connect(other_wire, SigSpec(wire, i));
 | |
| 							else
 | |
| 								module->connect(SigSpec(wire, i), other_wire);
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				module->fixup_ports();
 | |
| 				wideports_cache.clear();
 | |
| 
 | |
| 				if (run_clean)
 | |
| 				{
 | |
| 					Const buffer_lut(vector<RTLIL::State>({State::S0, State::S1}));
 | |
| 					vector<Cell*> remove_cells;
 | |
| 
 | |
| 					for (auto cell : module->cells())
 | |
| 						if (cell->type == ID($lut) && cell->getParam(ID::LUT) == buffer_lut) {
 | |
| 							module->connect(cell->getPort(ID::Y), cell->getPort(ID::A));
 | |
| 							remove_cells.push_back(cell);
 | |
| 						}
 | |
| 
 | |
| 					for (auto cell : remove_cells)
 | |
| 						module->remove(cell);
 | |
| 
 | |
| 					Wire *true_wire = module->wire(ID($true));
 | |
| 					Wire *false_wire = module->wire(ID($false));
 | |
| 					Wire *undef_wire = module->wire(ID($undef));
 | |
| 
 | |
| 					if (true_wire != nullptr)
 | |
| 						module->rename(true_wire, stringf("$true$%d", ++blif_maxnum));
 | |
| 
 | |
| 					if (false_wire != nullptr)
 | |
| 						module->rename(false_wire, stringf("$false$%d", ++blif_maxnum));
 | |
| 
 | |
| 					if (undef_wire != nullptr)
 | |
| 						module->rename(undef_wire, stringf("$undef$%d", ++blif_maxnum));
 | |
| 
 | |
| 					autoidx = std::max(autoidx, blif_maxnum+1);
 | |
| 					blif_maxnum = 0;
 | |
| 				}
 | |
| 
 | |
| 				module = nullptr;
 | |
| 				lastcell = nullptr;
 | |
| 				obj_attributes = nullptr;
 | |
| 				obj_parameters = nullptr;
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".area") || !strcmp(cmd, ".delay") || !strcmp(cmd, ".wire_load_slope") || !strcmp(cmd, ".wire") ||
 | |
| 			    !strcmp(cmd, ".input_arrival") || !strcmp(cmd, ".default_input_arrival") || !strcmp(cmd, ".output_required") ||
 | |
| 			    !strcmp(cmd, ".default_output_required") || !strcmp(cmd, ".input_drive") || !strcmp(cmd, ".default_input_drive") ||
 | |
| 			    !strcmp(cmd, ".max_input_load") || !strcmp(cmd, ".default_max_input_load") || !strcmp(cmd, ".output_load") ||
 | |
| 			    !strcmp(cmd, ".default_output_load"))
 | |
| 			{
 | |
| 				log_warning("Blif delay constraints (%s) are not supported.", cmd);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".inputs") || !strcmp(cmd, ".outputs"))
 | |
| 			{
 | |
| 				char *p;
 | |
| 				while ((p = strtok(NULL, " \t\r\n")) != NULL)
 | |
| 				{
 | |
| 					RTLIL::IdString wire_name(stringf("\\%s", p));
 | |
| 					RTLIL::Wire *wire = module->wire(wire_name);
 | |
| 					if (wire == nullptr)
 | |
| 						wire = module->addWire(wire_name);
 | |
| 					if (!strcmp(cmd, ".inputs"))
 | |
| 						wire->port_input = true;
 | |
| 					else
 | |
| 						wire->port_output = true;
 | |
| 
 | |
| 					if (wideports) {
 | |
| 						std::pair<RTLIL::IdString, int> wp = wideports_split(p);
 | |
| 						if (!wp.first.empty() && wp.second >= 0) {
 | |
| 							wideports_cache[wp.first].first = std::max(wideports_cache[wp.first].first, wp.second + 1);
 | |
| 							wideports_cache[wp.first].second = !strcmp(cmd, ".inputs");
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				obj_attributes = nullptr;
 | |
| 				obj_parameters = nullptr;
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".cname"))
 | |
| 			{
 | |
| 				char *p = strtok(NULL, " \t\r\n");
 | |
| 				if (p == NULL)
 | |
| 					goto error;
 | |
| 
 | |
| 				if(lastcell == nullptr || module == nullptr)
 | |
| 				{
 | |
| 					err_reason = stringf("No primitive object to attach .cname %s.", p);
 | |
| 					goto error_with_reason;
 | |
| 				}
 | |
| 
 | |
| 				module->rename(lastcell, RTLIL::escape_id(p));
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".attr") || !strcmp(cmd, ".param")) {
 | |
| 				char *n = strtok(NULL, " \t\r\n");
 | |
| 				char *v = strtok(NULL, "\r\n");
 | |
| 				IdString id_n = RTLIL::escape_id(n);
 | |
| 				Const const_v;
 | |
| 				if (v[0] == '"') {
 | |
| 					std::string str(v+1);
 | |
| 					if (str.back() == '"')
 | |
| 						str.resize(str.size()-1);
 | |
| 					const_v = Const(str);
 | |
| 				} else {
 | |
| 					int n = strlen(v);
 | |
| 					Const::Builder const_v_builder(n);
 | |
| 					for (int i = 0; i < n; i++)
 | |
| 						const_v_builder.push_back(v[n-i-1] != '0' ? State::S1 : State::S0);
 | |
| 					const_v = const_v_builder.build();
 | |
| 				}
 | |
| 				if (!strcmp(cmd, ".attr")) {
 | |
| 					if (obj_attributes == nullptr) {
 | |
| 						err_reason = stringf("No object to attach .attr too.");
 | |
| 						goto error_with_reason;
 | |
| 					}
 | |
| 					(*obj_attributes)[id_n] = const_v;
 | |
| 				} else {
 | |
| 					if (obj_parameters == nullptr) {
 | |
| 						err_reason = stringf("No object to attach .param too.");
 | |
| 						goto error_with_reason;
 | |
| 					}
 | |
| 					(*obj_parameters)[id_n] = const_v;
 | |
| 				}
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".latch"))
 | |
| 			{
 | |
| 				char *d = strtok(NULL, " \t\r\n");
 | |
| 				char *q = strtok(NULL, " \t\r\n");
 | |
| 				char *edge = strtok(NULL, " \t\r\n");
 | |
| 				char *clock = strtok(NULL, " \t\r\n");
 | |
| 				char *init = strtok(NULL, " \t\r\n");
 | |
| 				RTLIL::Cell *cell = nullptr;
 | |
| 
 | |
| 				if (clock == nullptr && edge != nullptr) {
 | |
| 					init = edge;
 | |
| 					edge = nullptr;
 | |
| 				}
 | |
| 
 | |
| 				if (init != nullptr && (init[0] == '0' || init[0] == '1'))
 | |
| 					blif_wire(q)->attributes[ID::init] = Const(init[0] == '1' ? 1 : 0, 1);
 | |
| 
 | |
| 				if (clock == nullptr)
 | |
| 					goto no_latch_clock;
 | |
| 
 | |
| 				if (!strcmp(edge, "re"))
 | |
| 					cell = module->addDffGate(NEW_ID, blif_wire(clock), blif_wire(d), blif_wire(q));
 | |
| 				else if (!strcmp(edge, "fe"))
 | |
| 					cell = module->addDffGate(NEW_ID, blif_wire(clock), blif_wire(d), blif_wire(q), false);
 | |
| 				else if (!strcmp(edge, "ah"))
 | |
| 					cell = module->addDlatchGate(NEW_ID, blif_wire(clock), blif_wire(d), blif_wire(q));
 | |
| 				else if (!strcmp(edge, "al"))
 | |
| 					cell = module->addDlatchGate(NEW_ID, blif_wire(clock), blif_wire(d), blif_wire(q), false);
 | |
| 				else {
 | |
| 			no_latch_clock:
 | |
| 					if (dff_name.empty()) {
 | |
| 						cell = module->addFfGate(NEW_ID, blif_wire(d), blif_wire(q));
 | |
| 					} else {
 | |
| 						cell = module->addCell(NEW_ID, dff_name);
 | |
| 						cell->setPort(ID::D, blif_wire(d));
 | |
| 						cell->setPort(ID::Q, blif_wire(q));
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				lastcell = cell;
 | |
| 				obj_attributes = &cell->attributes;
 | |
| 				obj_parameters = &cell->parameters;
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".gate") || !strcmp(cmd, ".subckt"))
 | |
| 			{
 | |
| 				char *p = strtok(NULL, " \t\r\n");
 | |
| 				if (p == NULL)
 | |
| 					goto error;
 | |
| 
 | |
| 				IdString celltype = RTLIL::escape_id(p);
 | |
| 				RTLIL::Cell *cell = module->addCell(NEW_ID, celltype);
 | |
| 				RTLIL::Module *cell_mod = design->module(celltype);
 | |
| 
 | |
| 				dict<RTLIL::IdString, dict<int, SigBit>> cell_wideports_cache;
 | |
| 
 | |
| 				while ((p = strtok(NULL, " \t\r\n")) != NULL)
 | |
| 				{
 | |
| 					char *q = strchr(p, '=');
 | |
| 					if (q == NULL || !q[0])
 | |
| 						goto error;
 | |
| 					*(q++) = 0;
 | |
| 
 | |
| 					if (wideports) {
 | |
| 						std::pair<RTLIL::IdString, int> wp = wideports_split(p);
 | |
| 						if (wp.first.empty())
 | |
| 							cell->setPort(RTLIL::escape_id(p), *q ? blif_wire(q) : SigSpec());
 | |
| 						else
 | |
| 							cell_wideports_cache[wp.first][wp.second] = blif_wire(q);
 | |
| 					} else {
 | |
| 						cell->setPort(RTLIL::escape_id(p), *q ? blif_wire(q) : SigSpec());
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				for (auto &it : cell_wideports_cache)
 | |
| 				{
 | |
| 					int width = 0;
 | |
| 					int offset = 0;
 | |
| 					bool upto = false;
 | |
| 					for (auto &b : it.second)
 | |
| 						width = std::max(width, b.first + 1);
 | |
| 
 | |
| 					if (cell_mod) {
 | |
| 						Wire *cell_port = cell_mod->wire(it.first);
 | |
| 						if (cell_port && (cell_port->port_input || cell_port->port_output)) {
 | |
| 							offset = cell_port->start_offset;
 | |
| 							upto = cell_port->upto;
 | |
| 							width = cell_port->width;
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					SigSpec sig;
 | |
| 
 | |
| 					for (int i = 0; i < width; i++) {
 | |
| 						int idx = offset + (upto ? width - 1 - i: i);
 | |
| 						if (it.second.count(idx))
 | |
| 							sig.append(it.second.at(idx));
 | |
| 						else
 | |
| 							sig.append(module->addWire(NEW_ID));
 | |
| 					}
 | |
| 
 | |
| 					cell->setPort(it.first, sig);
 | |
| 				}
 | |
| 
 | |
| 				lastcell = cell;
 | |
| 				obj_attributes = &cell->attributes;
 | |
| 				obj_parameters = &cell->parameters;
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			obj_attributes = nullptr;
 | |
| 			obj_parameters = nullptr;
 | |
| 
 | |
| 			if (!strcmp(cmd, ".barbuf") || !strcmp(cmd, ".conn"))
 | |
| 			{
 | |
| 				char *p = strtok(NULL, " \t\r\n");
 | |
| 				if (p == NULL)
 | |
| 					goto error;
 | |
| 
 | |
| 				char *q = strtok(NULL, " \t\r\n");
 | |
| 				if (q == NULL)
 | |
| 					goto error;
 | |
| 
 | |
| 				module->connect(blif_wire(q), blif_wire(p));
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp(cmd, ".names"))
 | |
| 			{
 | |
| 				char *p;
 | |
| 				RTLIL::SigSpec input_sig, output_sig;
 | |
| 				while ((p = strtok(NULL, " \t\r\n")) != NULL)
 | |
| 					input_sig.append(blif_wire(p));
 | |
| 				output_sig = input_sig.extract(input_sig.size()-1, 1);
 | |
| 				input_sig = input_sig.extract(0, input_sig.size()-1);
 | |
| 
 | |
| 				if (input_sig.size() == 0)
 | |
| 				{
 | |
| 					RTLIL::State state = RTLIL::State::Sa;
 | |
| 					while (1) {
 | |
| 						if (!read_next_line(buffer, buffer_size, line_count, f))
 | |
| 							goto error;
 | |
| 						for (int i = 0; buffer[i]; i++) {
 | |
| 							if (buffer[i] == ' ' || buffer[i] == '\t')
 | |
| 								continue;
 | |
| 							if (i == 0 && buffer[i] == '.')
 | |
| 								goto finished_parsing_constval;
 | |
| 							if (buffer[i] == '0') {
 | |
| 								if (state == RTLIL::State::S1)
 | |
| 									goto error;
 | |
| 								state = RTLIL::State::S0;
 | |
| 								continue;
 | |
| 							}
 | |
| 							if (buffer[i] == '1') {
 | |
| 								if (state == RTLIL::State::S0)
 | |
| 									goto error;
 | |
| 								state = RTLIL::State::S1;
 | |
| 								continue;
 | |
| 							}
 | |
| 							goto error;
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 				finished_parsing_constval:
 | |
| 					if (state == RTLIL::State::Sa)
 | |
| 						state = RTLIL::State::S0;
 | |
| 					if (output_sig.as_wire()->name == ID($undef))
 | |
| 						state = RTLIL::State::Sx;
 | |
| 					module->connect(RTLIL::SigSig(output_sig, state));
 | |
| 					goto continue_without_read;
 | |
| 				}
 | |
| 
 | |
| 				if (sop_mode)
 | |
| 				{
 | |
| 					sopcell = module->addCell(NEW_ID, ID($sop));
 | |
| 					sopcell->parameters[ID::WIDTH] = RTLIL::Const(input_sig.size());
 | |
| 					sopcell->parameters[ID::DEPTH] = 0;
 | |
| 					sopcell->parameters[ID::TABLE] = RTLIL::Const();
 | |
| 					sopcell->setPort(ID::A, input_sig);
 | |
| 					sopcell->setPort(ID::Y, output_sig);
 | |
| 					sopmode = -1;
 | |
| 					lastcell = sopcell;
 | |
| 				}
 | |
| 				else if (input_sig.size() > lut_input_plane_limit)
 | |
| 				{
 | |
| 					err_reason = stringf("names' input plane must have fewer than %d signals.", lut_input_plane_limit + 1);
 | |
| 					goto error_with_reason;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					RTLIL::Cell *cell = module->addCell(NEW_ID, ID($lut));
 | |
| 					cell->parameters[ID::WIDTH] = RTLIL::Const(input_sig.size());
 | |
| 					cell->parameters[ID::LUT] = RTLIL::Const(RTLIL::State::Sx, 1 << input_sig.size());
 | |
| 					cell->setPort(ID::A, input_sig);
 | |
| 					cell->setPort(ID::Y, output_sig);
 | |
| 					lutptr = &cell->parameters.at(ID::LUT);
 | |
| 					lut_default_state = RTLIL::State::Sx;
 | |
| 					lastcell = cell;
 | |
| 				}
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			goto error;
 | |
| 		}
 | |
| 
 | |
| 		if (lutptr == NULL && sopcell == NULL)
 | |
| 			goto error;
 | |
| 
 | |
| 		char *input = strtok(buffer, " \t\r\n");
 | |
| 		char *output = strtok(NULL, " \t\r\n");
 | |
| 
 | |
| 		if (input == NULL || output == NULL || (strcmp(output, "0") && strcmp(output, "1")))
 | |
| 			goto error;
 | |
| 
 | |
| 		int input_len = strlen(input);
 | |
| 
 | |
| 		if (sopcell)
 | |
| 		{
 | |
| 			log_assert(sopcell->parameters[ID::WIDTH].as_int() == input_len);
 | |
| 			sopcell->parameters[ID::DEPTH] = sopcell->parameters[ID::DEPTH].as_int() + 1;
 | |
| 
 | |
| 			Const::Builder table_bits_builder(input_len * 2);
 | |
| 			for (int i = 0; i < input_len; i++)
 | |
| 				switch (input[i]) {
 | |
| 					case '0':
 | |
| 						table_bits_builder.push_back(State::S1);
 | |
| 						table_bits_builder.push_back(State::S0);
 | |
| 						break;
 | |
| 					case '1':
 | |
| 						table_bits_builder.push_back(State::S0);
 | |
| 						table_bits_builder.push_back(State::S1);
 | |
| 						break;
 | |
| 					default:
 | |
| 						table_bits_builder.push_back(State::S0);
 | |
| 						table_bits_builder.push_back(State::S0);
 | |
| 						break;
 | |
| 				}
 | |
| 			sopcell->parameters[ID::TABLE].append(table_bits_builder.build());
 | |
| 
 | |
| 			if (sopmode == -1) {
 | |
| 				sopmode = (*output == '1');
 | |
| 				if (!sopmode) {
 | |
| 					SigSpec outnet = sopcell->getPort(ID::Y);
 | |
| 					SigSpec tempnet = module->addWire(NEW_ID);
 | |
| 					module->addNotGate(NEW_ID, tempnet, outnet);
 | |
| 					sopcell->setPort(ID::Y, tempnet);
 | |
| 				}
 | |
| 			} else
 | |
| 				log_assert(sopmode == (*output == '1'));
 | |
| 		}
 | |
| 
 | |
| 		if (lutptr)
 | |
| 		{
 | |
| 			if (input_len > lut_input_plane_limit)
 | |
| 				goto error;
 | |
| 
 | |
| 			for (int i = 0; i < (1 << input_len); i++) {
 | |
| 				for (int j = 0; j < input_len; j++) {
 | |
| 					char c1 = input[j];
 | |
| 					if (c1 != '-') {
 | |
| 						char c2 = (i & (1 << j)) != 0 ? '1' : '0';
 | |
| 						if (c1 != c2)
 | |
| 							goto try_next_value;
 | |
| 					}
 | |
| 				}
 | |
| 				lutptr->set(i, !strcmp(output, "0") ? RTLIL::State::S0 : RTLIL::State::S1);
 | |
| 			try_next_value:;
 | |
| 			}
 | |
| 
 | |
| 			lut_default_state = !strcmp(output, "0") ? RTLIL::State::S1 : RTLIL::State::S0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return;
 | |
| 
 | |
| error:
 | |
| 	log_error("Syntax error in line %d!\n", line_count);
 | |
| error_with_reason:
 | |
| 	log_error("Syntax error in line %d: %s\n", line_count, err_reason);
 | |
| }
 | |
| 
 | |
| struct BlifFrontend : public Frontend {
 | |
| 	BlifFrontend() : Frontend("blif", "read BLIF file") { }
 | |
| 	void help() override
 | |
| 	{
 | |
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | |
| 		log("\n");
 | |
| 		log("    read_blif [options] [filename]\n");
 | |
| 		log("\n");
 | |
| 		log("Load modules from a BLIF file into the current design.\n");
 | |
| 		log("\n");
 | |
| 		log("    -sop\n");
 | |
| 		log("        Create $sop cells instead of $lut cells\n");
 | |
| 		log("\n");
 | |
| 		log("    -wideports\n");
 | |
| 		log("        Merge ports that match the pattern 'name[int]' into a single\n");
 | |
| 		log("        multi-bit port 'name'.\n");
 | |
| 		log("\n");
 | |
| 	}
 | |
| 	void execute(std::istream *&f, std::string filename, std::vector<std::string> args, RTLIL::Design *design) override
 | |
| 	{
 | |
| 		bool sop_mode = false;
 | |
| 		bool wideports = false;
 | |
| 
 | |
| 		log_header(design, "Executing BLIF frontend.\n");
 | |
| 
 | |
| 		size_t argidx;
 | |
| 		for (argidx = 1; argidx < args.size(); argidx++) {
 | |
| 			std::string arg = args[argidx];
 | |
| 			if (arg == "-sop") {
 | |
| 				sop_mode = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (arg == "-wideports") {
 | |
| 				wideports = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 		extra_args(f, filename, args, argidx);
 | |
| 
 | |
| 		parse_blif(design, *f, "", true, sop_mode, wideports);
 | |
| 	}
 | |
| } BlifFrontend;
 | |
| 
 | |
| YOSYS_NAMESPACE_END
 | |
| 
 |