2002-04-21 06:59:07 +02:00
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2002 Stephen Williams (steve@icarus.com)
|
|
|
|
|
*
|
|
|
|
|
* This source code is free software; you can redistribute it
|
|
|
|
|
* and/or modify it in source code form under the terms of the GNU
|
|
|
|
|
* General Public License as published by the Free Software
|
|
|
|
|
* Foundation; either version 2 of the License, or (at your option)
|
|
|
|
|
* any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
|
|
|
*/
|
2002-08-12 03:34:58 +02:00
|
|
|
#ifdef HAVE_CVS_IDENT
|
2005-07-11 18:56:50 +02:00
|
|
|
#ident "$Id: net_nex_input.cc,v 1.14 2005/07/11 16:56:50 steve Exp $"
|
2002-04-21 06:59:07 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
# include "config.h"
|
|
|
|
|
|
|
|
|
|
# include <iostream>
|
|
|
|
|
|
|
|
|
|
# include <cassert>
|
|
|
|
|
# include <typeinfo>
|
|
|
|
|
# include "netlist.h"
|
|
|
|
|
# include "netmisc.h"
|
|
|
|
|
|
|
|
|
|
NexusSet* NetExpr::nex_input()
|
|
|
|
|
{
|
|
|
|
|
cerr << get_line()
|
|
|
|
|
<< ": internal error: nex_input not implemented: "
|
|
|
|
|
<< *this << endl;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetProc::nex_input()
|
|
|
|
|
{
|
|
|
|
|
cerr << get_line()
|
|
|
|
|
<< ": internal error: NetProc::nex_input not implemented"
|
|
|
|
|
<< endl;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetEBinary::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = left_->nex_input();
|
|
|
|
|
NexusSet*tmp = right_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetEConcat::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = parms_[0]->nex_input();
|
|
|
|
|
for (unsigned idx = 1 ; idx < parms_.count() ; idx += 1) {
|
|
|
|
|
NexusSet*tmp = parms_[idx]->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* A constant has not inputs, so always return an empty set.
|
|
|
|
|
*/
|
|
|
|
|
NexusSet* NetEConst::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
2003-01-26 22:15:58 +01:00
|
|
|
NexusSet* NetECReal::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 06:59:07 +02:00
|
|
|
NexusSet* NetEMemory::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = idx_->nex_input();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* A parameter by definition has no inputs. It represents a constant
|
|
|
|
|
* value, even if that value is a constant expression.
|
|
|
|
|
*/
|
|
|
|
|
NexusSet* NetEParam::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
2003-04-22 06:48:29 +02:00
|
|
|
NexusSet* NetEEvent::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 06:59:07 +02:00
|
|
|
NexusSet* NetEScope::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetESelect::nex_input()
|
|
|
|
|
{
|
2003-07-26 05:34:42 +02:00
|
|
|
NexusSet*result = base_? base_->nex_input() : new NexusSet();
|
2002-04-21 06:59:07 +02:00
|
|
|
NexusSet*tmp = expr_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetESFunc::nex_input()
|
|
|
|
|
{
|
|
|
|
|
if (nparms_ == 0)
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
|
|
|
|
|
NexusSet*result = parms_[0]->nex_input();
|
|
|
|
|
for (unsigned idx = 1 ; idx < nparms_ ; idx += 1) {
|
|
|
|
|
NexusSet*tmp = parms_[idx]->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetESignal::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = new NexusSet;
|
|
|
|
|
for (unsigned idx = 0 ; idx < net_->pin_count() ; idx += 1)
|
|
|
|
|
result->add(net_->pin(idx).nexus());
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetETernary::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*tmp;
|
|
|
|
|
NexusSet*result = cond_->nex_input();
|
|
|
|
|
|
|
|
|
|
tmp = true_val_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
|
|
|
|
|
tmp = false_val_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetEUFunc::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = new NexusSet;
|
|
|
|
|
for (unsigned idx = 0 ; idx < parms_.count() ; idx += 1) {
|
|
|
|
|
NexusSet*tmp = parms_[idx]->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetEUnary::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return expr_->nex_input();
|
|
|
|
|
}
|
|
|
|
|
|
2004-09-04 06:24:15 +02:00
|
|
|
NexusSet* NetAssign_::nex_input()
|
|
|
|
|
{
|
|
|
|
|
if (bmux_ == 0)
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
else
|
|
|
|
|
return bmux_->nex_input();
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 19:43:12 +02:00
|
|
|
NexusSet* NetAssignBase::nex_input()
|
2002-04-21 06:59:07 +02:00
|
|
|
{
|
2002-04-21 19:43:12 +02:00
|
|
|
NexusSet*result = rval_->nex_input();
|
2004-09-04 06:24:15 +02:00
|
|
|
|
|
|
|
|
/* It is possible that the lval_ can hav nex_input values. In
|
|
|
|
|
particular, index expressions are statement inputs as well,
|
|
|
|
|
so should be addressed here. */
|
|
|
|
|
for (NetAssign_*cur = lval_ ; cur ; cur = cur->more) {
|
|
|
|
|
NexusSet*tmp = cur->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 19:43:12 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2002-08-19 00:07:16 +02:00
|
|
|
/*
|
|
|
|
|
* The nex_input of a begin/end block is the NexusSet of bits that the
|
|
|
|
|
* block reads from outside the block. That means it is the union of
|
|
|
|
|
* the nex_input for all the substatements.
|
|
|
|
|
*
|
|
|
|
|
* The input set for a sequential set is not exactly the union of the
|
|
|
|
|
* input sets because there is the possibility of intermediate values,
|
|
|
|
|
* that don't deserve to be in the input set. To wit:
|
|
|
|
|
*
|
|
|
|
|
* begin
|
|
|
|
|
* t = a + b;
|
|
|
|
|
* c = ~t;
|
|
|
|
|
* end
|
|
|
|
|
*
|
|
|
|
|
* In this example, "t" should not be in the input set because it is
|
|
|
|
|
* used by the sequence as a temporary value.
|
|
|
|
|
*/
|
2002-04-21 19:43:12 +02:00
|
|
|
NexusSet* NetBlock::nex_input()
|
|
|
|
|
{
|
|
|
|
|
if (last_ == 0)
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
|
2002-08-19 00:07:16 +02:00
|
|
|
if (type_ == PARA) {
|
|
|
|
|
cerr << get_line() << ": internal error: Sorry, "
|
|
|
|
|
<< "I don't know how to synthesize fork/join blocks."
|
|
|
|
|
<< endl;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NetProc*cur = last_->next_;
|
|
|
|
|
/* This is the accumulated input set. */
|
|
|
|
|
NexusSet*result = new NexusSet;
|
|
|
|
|
/* This is an accumulated output set. */
|
|
|
|
|
NexusSet*prev = new NexusSet;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
/* Get the inputs for the current statement. */
|
2002-04-21 19:43:12 +02:00
|
|
|
NexusSet*tmp = cur->nex_input();
|
2002-08-19 00:07:16 +02:00
|
|
|
|
|
|
|
|
/* Remove from the input set those bits that are outputs
|
|
|
|
|
from previous statements. They aren't really inputs
|
|
|
|
|
to the block, just internal intermediate values. */
|
|
|
|
|
tmp->rem(*prev);
|
|
|
|
|
|
|
|
|
|
/* Add the corrected set to the accumulated input set. */
|
2002-04-21 19:43:12 +02:00
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
2002-08-19 00:07:16 +02:00
|
|
|
|
|
|
|
|
/* Add the current outputs to the accumulated output
|
|
|
|
|
set, for processing of later statements. */
|
|
|
|
|
cur->nex_output(*prev);
|
|
|
|
|
|
2002-07-15 01:47:58 +02:00
|
|
|
cur = cur->next_;
|
2002-08-19 00:07:16 +02:00
|
|
|
} while (cur != last_->next_);
|
2002-04-21 19:43:12 +02:00
|
|
|
|
2002-04-21 06:59:07 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* The inputs to a case statement are the inputs to the expression,
|
|
|
|
|
* the inputs to all the guards, and the inputs to all the guarded
|
|
|
|
|
* statements.
|
|
|
|
|
*/
|
|
|
|
|
NexusSet* NetCase::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = expr_->nex_input();
|
|
|
|
|
if (result == 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
for (unsigned idx = 0 ; idx < nitems_ ; idx += 1) {
|
|
|
|
|
|
2003-10-26 05:50:46 +01:00
|
|
|
/* Skip cases that have empty statements. */
|
|
|
|
|
if (items_[idx].statement == 0)
|
|
|
|
|
continue;
|
|
|
|
|
|
2002-04-21 06:59:07 +02:00
|
|
|
NexusSet*tmp = items_[idx].statement->nex_input();
|
|
|
|
|
assert(tmp);
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
|
2002-11-16 06:45:41 +01:00
|
|
|
/* Usually, this is the guard expression. The default
|
|
|
|
|
case is special and is identified by a null
|
|
|
|
|
guard. The default guard obviously has no input. */
|
|
|
|
|
if (items_[idx].guard) {
|
|
|
|
|
tmp = items_[idx].guard->nex_input();
|
|
|
|
|
assert(tmp);
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
2002-04-21 06:59:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 19:43:12 +02:00
|
|
|
NexusSet* NetCAssign::nex_input()
|
|
|
|
|
{
|
|
|
|
|
cerr << get_line() << ": internal warning: NetCAssign::nex_input()"
|
|
|
|
|
<< " not implemented." << endl;
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetCondit::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = expr_->nex_input();
|
|
|
|
|
if (if_ != 0) {
|
|
|
|
|
NexusSet*tmp = if_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (else_ != 0) {
|
|
|
|
|
NexusSet*tmp = else_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetForce::nex_input()
|
|
|
|
|
{
|
|
|
|
|
cerr << get_line() << ": internal warning: NetForce::nex_input()"
|
|
|
|
|
<< " not implemented." << endl;
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetForever::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = statement_->nex_input();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* The NetPDelay statement is a statement of the form
|
|
|
|
|
*
|
|
|
|
|
* #<expr> <statement>
|
|
|
|
|
*
|
|
|
|
|
* The nex_input set is the input set of the <statement>. Do *not*
|
|
|
|
|
* include the input set of the <expr> because it does not affect the
|
|
|
|
|
* result.
|
|
|
|
|
*/
|
|
|
|
|
NexusSet* NetPDelay::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = statement_->nex_input();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetRepeat::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = statement_->nex_input();
|
|
|
|
|
NexusSet*tmp = expr_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetSTask::nex_input()
|
|
|
|
|
{
|
|
|
|
|
if (parms_.count() == 0)
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
|
|
|
|
|
NexusSet*result = parms_[0]->nex_input();
|
|
|
|
|
for (unsigned idx = 1 ; idx < parms_.count() ; idx += 1) {
|
|
|
|
|
NexusSet*tmp = parms_[idx]->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* The NetUTask represents a call to a user defined task. There are no
|
|
|
|
|
* parameters to consider, because the compiler already removed them
|
|
|
|
|
* and converted them to blocking assignments.
|
|
|
|
|
*/
|
|
|
|
|
NexusSet* NetUTask::nex_input()
|
|
|
|
|
{
|
|
|
|
|
return new NexusSet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NexusSet* NetWhile::nex_input()
|
|
|
|
|
{
|
|
|
|
|
NexusSet*result = proc_->nex_input();
|
|
|
|
|
NexusSet*tmp = cond_->nex_input();
|
|
|
|
|
result->add(*tmp);
|
|
|
|
|
delete tmp;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-21 06:59:07 +02:00
|
|
|
/*
|
|
|
|
|
* $Log: net_nex_input.cc,v $
|
2005-07-11 18:56:50 +02:00
|
|
|
* Revision 1.14 2005/07/11 16:56:50 steve
|
|
|
|
|
* Remove NetVariable and ivl_variable_t structures.
|
|
|
|
|
*
|
2005-01-24 06:28:30 +01:00
|
|
|
* Revision 1.13 2005/01/24 05:28:30 steve
|
|
|
|
|
* Remove the NetEBitSel and combine all bit/part select
|
|
|
|
|
* behavior into the NetESelect node and IVL_EX_SELECT
|
|
|
|
|
* ivl_target expression type.
|
|
|
|
|
*
|
2004-09-04 06:24:15 +02:00
|
|
|
* Revision 1.12 2004/09/04 04:24:15 steve
|
|
|
|
|
* PR1026: assignment statements can have sensitivities in the l-values.
|
|
|
|
|
*
|
2003-10-26 05:50:46 +01:00
|
|
|
* Revision 1.11 2003/10/26 04:50:46 steve
|
|
|
|
|
* Case with empty statements has no inputs.
|
|
|
|
|
*
|
2003-07-26 05:34:42 +02:00
|
|
|
* Revision 1.10 2003/07/26 03:34:42 steve
|
|
|
|
|
* Start handling pad of expressions in code generators.
|
|
|
|
|
*
|
2003-04-22 06:48:29 +02:00
|
|
|
* Revision 1.9 2003/04/22 04:48:29 steve
|
|
|
|
|
* Support event names as expressions elements.
|
|
|
|
|
*
|
2003-01-26 22:15:58 +01:00
|
|
|
* Revision 1.8 2003/01/26 21:15:58 steve
|
|
|
|
|
* Rework expression parsing and elaboration to
|
|
|
|
|
* accommodate real/realtime values and expressions.
|
|
|
|
|
*
|
2002-11-16 06:45:41 +01:00
|
|
|
* Revision 1.7 2002/11/16 05:45:41 steve
|
|
|
|
|
* Handle default: case in net_inputs for NetCase.
|
|
|
|
|
*
|
2002-08-19 00:07:16 +02:00
|
|
|
* Revision 1.6 2002/08/18 22:07:16 steve
|
|
|
|
|
* Detect temporaries in sequential block synthesis.
|
|
|
|
|
*
|
2002-08-12 03:34:58 +02:00
|
|
|
* Revision 1.5 2002/08/12 01:34:59 steve
|
|
|
|
|
* conditional ident string using autoconfig.
|
|
|
|
|
*
|
2002-07-15 01:47:58 +02:00
|
|
|
* Revision 1.4 2002/07/14 23:47:58 steve
|
|
|
|
|
* Infinite loop in nex_input of NetBlock objects.
|
|
|
|
|
*
|
2002-06-05 05:44:25 +02:00
|
|
|
* Revision 1.3 2002/06/05 03:44:25 steve
|
|
|
|
|
* Add support for memory words in l-value of
|
|
|
|
|
* non-blocking assignments, and remove the special
|
|
|
|
|
* NetAssignMem_ and NetAssignMemNB classes.
|
|
|
|
|
*
|
2002-04-21 19:43:12 +02:00
|
|
|
* Revision 1.2 2002/04/21 17:43:12 steve
|
|
|
|
|
* implement nex_input for behavioral statements.
|
|
|
|
|
*
|
2002-04-21 06:59:07 +02:00
|
|
|
* Revision 1.1 2002/04/21 04:59:08 steve
|
|
|
|
|
* Add support for conbinational events by finding
|
|
|
|
|
* the inputs to expressions and some statements.
|
|
|
|
|
* Get case and assignment statements working.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|