mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-25 08:54:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			256 lines
		
	
	
	
		
			7.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
	
		
			7.5 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.
 | |
|  *
 | |
|  *  ---
 | |
|  *
 | |
|  *  The Verilog frontend.
 | |
|  *
 | |
|  *  This frontend is using the AST frontend library (see frontends/ast/).
 | |
|  *  Thus this frontend does not generate RTLIL code directly but creates an
 | |
|  *  AST directly from the Verilog parse tree and then passes this AST to
 | |
|  *  the AST frontend library.
 | |
|  *
 | |
|  *  ---
 | |
|  *
 | |
|  *  This file contains an ad-hoc parser for Verilog constants. The Verilog
 | |
|  *  lexer does only recognize a constant but does not actually split it to its
 | |
|  *  components. I.e. it just passes the Verilog code for the constant to the
 | |
|  *  bison parser. The parser then uses the function const2ast() from this file
 | |
|  *  to create an AST node for the constant.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include "verilog_frontend.h"
 | |
| #include "kernel/log.h"
 | |
| #include <string.h>
 | |
| #include <math.h>
 | |
| 
 | |
| YOSYS_NAMESPACE_BEGIN
 | |
| 
 | |
| using namespace AST;
 | |
| using namespace VERILOG_FRONTEND;
 | |
| 
 | |
| void ConstParser::log_maybe_loc_error(std::string msg) {
 | |
| 	log_file_error(*loc.begin.filename, loc.begin.line, "%s", msg);
 | |
| }
 | |
| 
 | |
| void ConstParser::log_maybe_loc_warn(std::string msg) {
 | |
| 	log_file_warning(*loc.begin.filename, loc.begin.line, "%s", msg);
 | |
| }
 | |
| 
 | |
| // divide an arbitrary length decimal number by two and return the rest
 | |
| int ConstParser::my_decimal_div_by_two(std::vector<uint8_t> &digits)
 | |
| {
 | |
| 	int carry = 0;
 | |
| 	for (size_t i = 0; i < digits.size(); i++) {
 | |
| 		if (digits[i] >= 10)
 | |
| 			log_maybe_loc_error("Invalid use of [a-fxz?] in decimal constant.\n");
 | |
| 		digits[i] += carry * 10;
 | |
| 		carry = digits[i] % 2;
 | |
| 		digits[i] /= 2;
 | |
| 	}
 | |
| 	while (!digits.empty() && !digits.front())
 | |
| 		digits.erase(digits.begin());
 | |
| 	return carry;
 | |
| }
 | |
| 
 | |
| // find the number of significant bits in a binary number (not including the sign bit)
 | |
| int ConstParser::my_ilog2(int x)
 | |
| {
 | |
| 	int ret = 0;
 | |
| 	while (x != 0 && x != -1) {
 | |
| 		x = x >> 1;
 | |
| 		ret++;
 | |
| 	}
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| // parse a binary, decimal, hexadecimal or octal number with support for special bits ('x', 'z' and '?')
 | |
| void ConstParser::my_strtobin(std::vector<RTLIL::State> &data, const char *str, int len_in_bits, int base, char case_type, bool is_unsized)
 | |
| {
 | |
| 	// all digits in string (MSB at index 0)
 | |
| 	std::vector<uint8_t> digits;
 | |
| 
 | |
| 	while (*str) {
 | |
| 		if ('0' <= *str && *str <= '9')
 | |
| 			digits.push_back(*str - '0');
 | |
| 		else if ('a' <= *str && *str <= 'f')
 | |
| 			digits.push_back(10 + *str - 'a');
 | |
| 		else if ('A' <= *str && *str <= 'F')
 | |
| 			digits.push_back(10 + *str - 'A');
 | |
| 		else if (*str == 'x' || *str == 'X')
 | |
| 			digits.push_back(0xf0);
 | |
| 		else if (*str == 'z' || *str == 'Z' || *str == '?')
 | |
| 			digits.push_back(0xf1);
 | |
| 		str++;
 | |
| 	}
 | |
| 
 | |
| 	if (base == 10 && GetSize(digits) == 1 && digits.front() >= 0xf0)
 | |
| 		base = 2;
 | |
| 
 | |
| 	data.clear();
 | |
| 
 | |
| 	if (base == 10) {
 | |
| 		while (!digits.empty())
 | |
| 			data.push_back(my_decimal_div_by_two(digits) ? State::S1 : State::S0);
 | |
| 	} else {
 | |
| 		int bits_per_digit = my_ilog2(base-1);
 | |
| 		for (auto it = digits.rbegin(), e = digits.rend(); it != e; it++) {
 | |
| 			if (*it > (base-1) && *it < 0xf0)
 | |
| 				log_maybe_loc_error(stringf("Digit larger than %d used in in base-%d constant.\n",
 | |
| 							base-1, base));
 | |
| 			for (int i = 0; i < bits_per_digit; i++) {
 | |
| 				int bitmask = 1 << i;
 | |
| 				if (*it == 0xf0)
 | |
| 					data.push_back(case_type == 'x' ? RTLIL::Sa : RTLIL::Sx);
 | |
| 				else if (*it == 0xf1)
 | |
| 					data.push_back(case_type == 'x' || case_type == 'z' ? RTLIL::Sa : RTLIL::Sz);
 | |
| 				else
 | |
| 					data.push_back((*it & bitmask) ? State::S1 : State::S0);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	int len = GetSize(data);
 | |
| 	RTLIL::State msb = data.empty() ? State::S0 : data.back();
 | |
| 
 | |
| 	if (len_in_bits < 0) {
 | |
| 		if (len < 32)
 | |
| 			data.resize(32, msb == State::S0 || msb == State::S1 ? RTLIL::S0 : msb);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (is_unsized && (len > len_in_bits))
 | |
| 		log_maybe_loc_error(stringf("Unsized constant must have width of 1 bit, but have %d bits!\n", len));
 | |
| 
 | |
| 	for (len = len - 1; len >= 0; len--)
 | |
| 		if (data[len] == State::S1)
 | |
| 			break;
 | |
| 	if (msb == State::S0 || msb == State::S1) {
 | |
| 		len += 1;
 | |
| 		data.resize(len_in_bits, State::S0);
 | |
| 	} else {
 | |
| 		len += 2;
 | |
| 		data.resize(len_in_bits, msb);
 | |
| 	}
 | |
| 
 | |
| 	if (len_in_bits == 0)
 | |
| 		log_maybe_loc_error("Illegal integer constant size of zero (IEEE 1800-2012, 5.7).\n");
 | |
| 
 | |
| 	if (len > len_in_bits)
 | |
| 		log_maybe_loc_warn(stringf("Literal has a width of %d bit, but value requires %d bit.\n",
 | |
| 			len_in_bits, len));
 | |
| }
 | |
| // convert the Verilog code for a constant to an AST node
 | |
| std::unique_ptr<AstNode> ConstParser::const2ast(std::string code, char case_type, bool warn_z)
 | |
| {
 | |
| 	if (warn_z) {
 | |
| 		auto ret = const2ast(code, case_type);
 | |
| 		if (ret != nullptr && std::find(ret->bits.begin(), ret->bits.end(), RTLIL::State::Sz) != ret->bits.end())
 | |
| 			log_maybe_loc_warn("Yosys has only limited support for tri-state logic at the moment.\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	const char *str = code.c_str();
 | |
| 
 | |
| 	// Strings
 | |
| 	if (*str == '"') {
 | |
| 		int len = strlen(str) - 2;
 | |
| 		std::vector<RTLIL::State> data;
 | |
| 		data.reserve(len * 8);
 | |
| 		for (int i = 0; i < len; i++) {
 | |
| 			unsigned char ch = str[len - i];
 | |
| 			for (int j = 0; j < 8; j++) {
 | |
| 				data.push_back((ch & 1) ? State::S1 : State::S0);
 | |
| 				ch = ch >> 1;
 | |
| 			}
 | |
| 		}
 | |
| 		auto ast = AstNode::mkconst_bits(loc, data, false);
 | |
| 		ast->str = code;
 | |
| 		return ast;
 | |
| 	}
 | |
| 
 | |
| 	for (size_t i = 0; i < code.size(); i++)
 | |
| 		if (code[i] == '_' || code[i] == ' ' || code[i] == '\t' || code[i] == '\r' || code[i] == '\n')
 | |
| 			code.erase(code.begin()+(i--));
 | |
| 	str = code.c_str();
 | |
| 
 | |
| 	char *endptr;
 | |
| 	long len_in_bits = strtol(str, &endptr, 10);
 | |
| 
 | |
| 	// Simple base-10 integer
 | |
| 	if (*endptr == 0) {
 | |
| 		std::vector<RTLIL::State> data;
 | |
| 		my_strtobin(data, str, -1, 10, case_type, false);
 | |
| 		if (data.back() == State::S1)
 | |
| 			data.push_back(State::S0);
 | |
| 		return AstNode::mkconst_bits(loc, data, true);
 | |
| 	}
 | |
| 
 | |
| 	// unsized constant
 | |
| 	if (str == endptr)
 | |
| 		len_in_bits = -1;
 | |
| 
 | |
| 	// The "<bits>'[sS]?[bodhBODH]<digits>" syntax
 | |
| 	if (*endptr == '\'')
 | |
| 	{
 | |
| 		std::vector<RTLIL::State> data;
 | |
| 		bool is_signed = false;
 | |
| 		bool is_unsized = len_in_bits < 0;
 | |
| 		if (*(endptr+1) == 's' || *(endptr+1) == 'S') {
 | |
| 			is_signed = true;
 | |
| 			endptr++;
 | |
| 		}
 | |
| 		switch (*(endptr+1))
 | |
| 		{
 | |
| 		case 'b':
 | |
| 		case 'B':
 | |
| 			my_strtobin(data, endptr+2, len_in_bits, 2, case_type, is_unsized);
 | |
| 			break;
 | |
| 		case 'o':
 | |
| 		case 'O':
 | |
| 			my_strtobin(data, endptr+2, len_in_bits, 8, case_type, is_unsized);
 | |
| 			break;
 | |
| 		case 'd':
 | |
| 		case 'D':
 | |
| 			my_strtobin(data, endptr+2, len_in_bits, 10, case_type, is_unsized);
 | |
| 			break;
 | |
| 		case 'h':
 | |
| 		case 'H':
 | |
| 			my_strtobin(data, endptr+2, len_in_bits, 16, case_type, is_unsized);
 | |
| 			break;
 | |
| 		default:
 | |
| 			char next_char = char(tolower(*(endptr+1)));
 | |
| 			if (next_char == '0' || next_char == '1' || next_char == 'x' || next_char == 'z') {
 | |
| 				is_unsized = true;
 | |
| 				my_strtobin(data, endptr+1, 1, 2, case_type, is_unsized);
 | |
| 			} else {
 | |
| 				return NULL;
 | |
| 			}
 | |
| 		}
 | |
| 		if (len_in_bits < 0) {
 | |
| 			if (is_signed && data.back() == State::S1)
 | |
| 				data.push_back(State::S0);
 | |
| 		}
 | |
| 		return AstNode::mkconst_bits(loc, data, is_signed, is_unsized);
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| YOSYS_NAMESPACE_END
 |