diff options
Diffstat (limited to 'frontends')
| -rw-r--r-- | frontends/ast/ast.cc | 1 | ||||
| -rw-r--r-- | frontends/ast/ast.h | 1 | ||||
| -rw-r--r-- | frontends/ast/genrtlil.cc | 24 | ||||
| -rw-r--r-- | frontends/ast/simplify.cc | 14 | ||||
| -rw-r--r-- | frontends/verific/verific.cc | 5 | ||||
| -rw-r--r-- | frontends/verilog/verilog_lexer.l | 2 | ||||
| -rw-r--r-- | frontends/verilog/verilog_parser.y | 33 | 
7 files changed, 73 insertions, 7 deletions
| diff --git a/frontends/ast/ast.cc b/frontends/ast/ast.cc index 03fd272da..9520ae32c 100644 --- a/frontends/ast/ast.cc +++ b/frontends/ast/ast.cc @@ -95,6 +95,7 @@ std::string AST::type2str(AstNodeType type)  	X(AST_TO_SIGNED)  	X(AST_TO_UNSIGNED)  	X(AST_SELFSZ) +	X(AST_CAST_SIZE)  	X(AST_CONCAT)  	X(AST_REPLICATE)  	X(AST_BIT_NOT) diff --git a/frontends/ast/ast.h b/frontends/ast/ast.h index 46864a4e1..9a5aa15f9 100644 --- a/frontends/ast/ast.h +++ b/frontends/ast/ast.h @@ -76,6 +76,7 @@ namespace AST  		AST_TO_SIGNED,  		AST_TO_UNSIGNED,  		AST_SELFSZ, +		AST_CAST_SIZE,  		AST_CONCAT,  		AST_REPLICATE,  		AST_BIT_NOT, diff --git a/frontends/ast/genrtlil.cc b/frontends/ast/genrtlil.cc index 9546558aa..e878d0dd2 100644 --- a/frontends/ast/genrtlil.cc +++ b/frontends/ast/genrtlil.cc @@ -814,6 +814,16 @@ void AstNode::detectSignWidthWorker(int &width_hint, bool &sign_hint, bool *foun  		children.at(0)->detectSignWidthWorker(sub_width_hint, sign_hint);  		break; +	case AST_CAST_SIZE: +		while (children.at(0)->simplify(true, false, false, 1, -1, false, false)) { } +		if (children.at(0)->type != AST_CONSTANT) +			log_file_error(filename, location.first_line, "Static cast with non constant expression!\n"); +		children.at(1)->detectSignWidthWorker(width_hint, sign_hint); +		width_hint = children.at(0)->bitsAsConst().as_int(); +		if (width_hint <= 0) +			log_file_error(filename, location.first_line, "Static cast with zero or negative size!\n"); +		break; +  	case AST_CONCAT:  		for (auto child : children) {  			sub_width_hint = 0; @@ -1289,6 +1299,20 @@ RTLIL::SigSpec AstNode::genRTLIL(int width_hint, bool sign_hint)  			return sig;  	} +	// changing the size of signal can be done directly using RTLIL::SigSpec +	case AST_CAST_SIZE: { +			RTLIL::SigSpec size = children[0]->genRTLIL(); +			RTLIL::SigSpec sig = children[1]->genRTLIL(); +			if (!size.is_fully_const()) +				log_file_error(filename, location.first_line, "Static cast with non constant expression!\n"); +			int width = size.as_int(); +			if (width <= 0) +				log_file_error(filename, location.first_line, "Static cast with zero or negative size!\n"); +			sig.extend_u0(width, sign_hint); +			is_signed = sign_hint; +			return sig; +		} +  	// concatenation of signals can be done directly using RTLIL::SigSpec  	case AST_CONCAT: {  			RTLIL::SigSpec sig; diff --git a/frontends/ast/simplify.cc b/frontends/ast/simplify.cc index 55e7da0aa..c4df5c0a0 100644 --- a/frontends/ast/simplify.cc +++ b/frontends/ast/simplify.cc @@ -950,6 +950,7 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,  	case AST_TO_SIGNED:  	case AST_TO_UNSIGNED:  	case AST_SELFSZ: +	case AST_CAST_SIZE:  	case AST_CONCAT:  	case AST_REPLICATE:  	case AST_REDUCE_AND: @@ -1126,6 +1127,10 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,  			bool in_param_here = in_param;  			if (i == 0 && (type == AST_REPLICATE || type == AST_WIRE))  				const_fold_here = true, in_param_here = true; +			if (i == 0 && (type == AST_GENIF || type == AST_GENCASE)) +				in_param_here = true; +			if (i == 1 && (type == AST_FOR || type == AST_GENFOR)) +				in_param_here = true;  			if (type == AST_PARAMETER || type == AST_LOCALPARAM)  				const_fold_here = true;  			if (i == 0 && (type == AST_ASSIGN || type == AST_ASSIGN_EQ || type == AST_ASSIGN_LE)) @@ -1942,7 +1947,7 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,  					continue;  				buf = child->clone(); -				while (buf->simplify(true, false, false, stage, width_hint, sign_hint, false)) { } +				while (buf->simplify(true, false, false, stage, width_hint, sign_hint, true)) { }  				if (buf->type != AST_CONSTANT) {  					// for (auto f : log_files)  					// 	dumpAst(f, "verilog-ast> "); @@ -3483,6 +3488,13 @@ replace_fcall_later:;  				}  			}  			break; +		case AST_CAST_SIZE: +			if (children.at(0)->type == AST_CONSTANT && children.at(1)->type == AST_CONSTANT) { +				int width = children[0]->bitsAsConst().as_int(); +				RTLIL::Const val = children[1]->bitsAsConst(width); +				newNode = mkconst_bits(val.bits, children[1]->is_signed); +			} +			break;  		case AST_CONCAT:  			string_op = !children.empty();  			for (auto it = children.begin(); it != children.end(); it++) { diff --git a/frontends/verific/verific.cc b/frontends/verific/verific.cc index ba631475f..9785b8eff 100644 --- a/frontends/verific/verific.cc +++ b/frontends/verific/verific.cc @@ -53,7 +53,7 @@ USING_YOSYS_NAMESPACE  #  error "Only Symbiotic EDA flavored Verific is supported. Please contact office@symbioticeda.com for commercial support for Yosys+Verific."  #endif -#if SYMBIOTIC_VERIFIC_API_VERSION < 1 +#if SYMBIOTIC_VERIFIC_API_VERSION < 202006  #  error "Please update your version of Symbiotic EDA flavored Verific."  #endif @@ -1111,7 +1111,8 @@ void VerificImporter::import_netlist(RTLIL::Design *design, Netlist *nl, std::se  			wire->start_offset = min(netbus->LeftIndex(), netbus->RightIndex());  			MapIter mibus;  			FOREACH_NET_OF_NETBUS(netbus, mibus, net) { -				import_attributes(wire->attributes, net, nl); +				if (net) +					import_attributes(wire->attributes, net, nl);  				break;  			} diff --git a/frontends/verilog/verilog_lexer.l b/frontends/verilog/verilog_lexer.l index 028106381..f2241066f 100644 --- a/frontends/verilog/verilog_lexer.l +++ b/frontends/verilog/verilog_lexer.l @@ -517,6 +517,8 @@ import[ \t\r\n]+\"(DPI|DPI-C)\"[ \t\r\n]+function[ \t\r\n]+ {  "<<<" { return OP_SSHL; }  ">>>" { return OP_SSHR; } +"'" { return OP_CAST; } +  "::"  { return TOK_PACKAGESEP; }  "++"  { return TOK_INCREMENT; }  "--"  { return TOK_DECREMENT; } diff --git a/frontends/verilog/verilog_parser.y b/frontends/verilog/verilog_parser.y index 96d9299fe..656910c0c 100644 --- a/frontends/verilog/verilog_parser.y +++ b/frontends/verilog/verilog_parser.y @@ -299,6 +299,7 @@ static void rewriteAsMemoryNode(AstNode *node, AstNode *rangeNode)  %left '+' '-'  %left '*' '/' '%'  %left OP_POW +%left OP_CAST  %right UNARY_OPS  %define parse.error verbose @@ -746,7 +747,7 @@ module_body:  module_body_stmt:  	task_func_decl | specify_block | param_decl | localparam_decl | typedef_decl | defparam_decl | specparam_declaration | wire_decl | assign_stmt | cell_stmt |  	enum_decl | struct_decl | -	always_stmt | TOK_GENERATE module_gen_body TOK_ENDGENERATE | defattr | assert_property | checker_decl | ignored_specify_block; +	always_stmt | TOK_GENERATE module_gen_body TOK_ENDGENERATE | defattr | assert_property | checker_decl | ignored_specify_block | /* empty statement */ ';';  checker_decl:  	TOK_CHECKER TOK_ID ';' { @@ -1330,6 +1331,8 @@ ignspec_id:  param_signed:  	TOK_SIGNED {  		astbuf1->is_signed = true; +	} | TOK_UNSIGNED { +		astbuf1->is_signed = false;  	} | /* empty */;  param_integer: @@ -1340,14 +1343,14 @@ param_integer:  		astbuf1->children.back()->children.push_back(AstNode::mkconst_int(31, true));  		astbuf1->children.back()->children.push_back(AstNode::mkconst_int(0, true));  		astbuf1->is_signed = true; -	} | /* empty */; +	}  param_real:  	TOK_REAL {  		if (astbuf1->children.size() != 1)  			frontend_verilog_yyerror("Parameter already declared as integer, cannot set to real.");  		astbuf1->children.push_back(new AstNode(AST_REALVALUE)); -	} | /* empty */; +	}  param_range:  	range { @@ -1358,8 +1361,12 @@ param_range:  		}  	}; +param_integer_type: param_integer param_signed +param_range_type: type_vec param_signed param_range +param_implicit_type: param_signed param_range +  param_type: -	param_signed param_integer param_real param_range | +	param_integer_type | param_real | param_range_type | param_implicit_type |  	hierarchical_type_id {  		astbuf1->is_custom_type = true;  		astbuf1->children.push_back(new AstNode(AST_WIRETYPE)); @@ -3042,6 +3049,24 @@ basic_expr:  		$$ = new AstNode(AST_LOGIC_NOT, $3);  		SET_AST_NODE_LOC($$, @1, @3);  		append_attr($$, $2); +	} | +	TOK_SIGNED OP_CAST '(' expr ')' { +		if (!sv_mode) +			frontend_verilog_yyerror("Static cast is only supported in SystemVerilog mode."); +		$$ = new AstNode(AST_TO_SIGNED, $4); +		SET_AST_NODE_LOC($$, @1, @4); +	} | +	TOK_UNSIGNED OP_CAST '(' expr ')' { +		if (!sv_mode) +			frontend_verilog_yyerror("Static cast is only supported in SystemVerilog mode."); +		$$ = new AstNode(AST_TO_UNSIGNED, $4); +		SET_AST_NODE_LOC($$, @1, @4); +	} | +	basic_expr OP_CAST '(' expr ')' { +		if (!sv_mode) +			frontend_verilog_yyerror("Static cast is only supported in SystemVerilog mode."); +		$$ = new AstNode(AST_CAST_SIZE, $1, $4); +		SET_AST_NODE_LOC($$, @1, @4);  	};  concat_list: | 
