iverilog/pform.cc

1024 lines
28 KiB
C++
Raw Normal View History

1998-11-04 00:28:49 +01:00
/*
* Copyright (c) 1998-2000 Stephen Williams (steve@icarus.com)
1998-11-04 00:28:49 +01:00
*
* 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
*/
#if !defined(WINNT) && !defined(macintosh)
2000-07-29 19:58:20 +02:00
#ident "$Id: pform.cc,v 1.63 2000/07/29 17:58:21 steve Exp $"
1998-11-04 00:28:49 +01:00
#endif
# include "compiler.h"
1998-11-04 00:28:49 +01:00
# include "pform.h"
# include "parse_misc.h"
2000-04-01 21:31:57 +02:00
# include "PEvent.h"
# include "PUdp.h"
1998-11-04 00:28:49 +01:00
# include <list>
# include <map>
1998-11-04 00:28:49 +01:00
# include <assert.h>
# include <typeinfo>
# include <strstream>
1998-11-04 00:28:49 +01:00
1998-12-09 05:02:47 +01:00
/*
* The lexor accesses the vl_* variables.
*/
string vl_file = "";
1998-11-04 00:28:49 +01:00
extern int VLparse();
1999-07-03 04:12:51 +02:00
static Module*pform_cur_module = 0;
static int pform_time_unit = 0;
static int pform_time_prec = 0;
1998-11-04 00:28:49 +01:00
/*
* The scope stack and the following functions handle the processing
* of scope. As I enter a scope, the push function is called, and as I
* leave a scope the pop function is called.
*
* The top module is not included in the scope list.
*/
struct scope_name_t {
string name;
struct scope_name_t*next;
};
static scope_name_t*scope_stack = 0;
void pform_set_timescale(int unit, int prec)
{
assert(unit >= prec);
pform_time_unit = unit;
pform_time_prec = prec;
}
void pform_push_scope(const string&name)
{
scope_name_t*cur = new scope_name_t;
cur->name = name;
cur->next = scope_stack;
scope_stack = cur;
}
void pform_pop_scope()
{
assert(scope_stack);
scope_name_t*cur = scope_stack;
scope_stack = cur->next;
delete cur;
}
static string scoped_name(string name)
{
scope_name_t*cur = scope_stack;
while (cur) {
name = cur->name + "." + name;
cur = cur->next;
}
return name;
}
static map<string,Module*> vl_modules;
static map<string,PUdp*> vl_primitives;
1998-11-04 00:28:49 +01:00
/*
* This function evaluates delay expressions. The result should be a
* simple constant that I can interpret as an unsigned number.
*/
static unsigned long evaluate_delay(PExpr*delay)
{
PENumber*pp = dynamic_cast<PENumber*>(delay);
if (pp == 0) {
VLerror("Sorry, delay expression is too complicated.");
return 0;
}
return pp->value().as_ulong();
}
void pform_startmodule(const string&name, svector<Module::port_t*>*ports)
1998-11-04 00:28:49 +01:00
{
1999-07-03 04:12:51 +02:00
assert( pform_cur_module == 0 );
1999-09-17 04:06:25 +02:00
/* The parser parses ``module foo()'' as having one
unconnected port, but it is really a module with no
ports. Fix it up here. */
if (ports && (ports->count() == 1) && ((*ports)[0] == 0)) {
delete ports;
ports = 0;
}
pform_cur_module = new Module(name, ports);
pform_cur_module->time_unit = pform_time_unit;
pform_cur_module->time_precision = pform_time_prec;
delete ports;
1998-11-04 00:28:49 +01:00
}
void pform_endmodule(const string&name)
{
1999-07-03 04:12:51 +02:00
assert(pform_cur_module);
assert(name == pform_cur_module->get_name());
vl_modules[name] = pform_cur_module;
pform_cur_module = 0;
1998-11-04 00:28:49 +01:00
}
bool pform_expression_is_constant(const PExpr*ex)
{
1999-07-03 04:12:51 +02:00
return ex->is_constant(pform_cur_module);
}
2000-07-29 19:58:20 +02:00
MIN_TYP_MAX min_typ_max_flag = TYP;
unsigned min_typ_max_warn = 10;
PExpr* pform_select_mtm_expr(PExpr*min, PExpr*typ, PExpr*max)
{
PExpr*res = 0;
switch (min_typ_max_flag) {
case MIN:
res = min;
delete typ;
delete max;
break;
case TYP:
res = typ;
delete min;
delete max;
break;
case MAX:
res = max;
delete min;
delete max;
break;
}
if (min_typ_max_warn > 0) {
cerr << res->get_line() << ": warning: choosing ";
switch (min_typ_max_flag) {
case MIN:
cerr << "min";
break;
case TYP:
cerr << "typ";
break;
case MAX:
cerr << "max";
break;
}
cerr << " expression." << endl;
min_typ_max_warn -= 1;
}
return res;
}
1999-07-10 03:03:18 +02:00
void pform_make_udp(const char*name, list<string>*parms,
1999-06-12 22:35:27 +02:00
svector<PWire*>*decl, list<string>*table,
Statement*init_expr)
{
assert(parms->size() > 0);
/* Put the declarations into a map, so that I can check them
off with the parameters in the list. If the port is already
in the map, merge the port type. I will rebuild a list
of parameters for the PUdp object. */
map<string,PWire*> defs;
for (unsigned idx = 0 ; idx < decl->count() ; idx += 1) {
string pname = (*decl)[idx]->name();
PWire*cur = defs[pname];
if (PWire*cur = defs[pname]) {
1999-06-21 03:02:16 +02:00
bool rc = true;
assert((*decl)[idx]);
if ((*decl)[idx]->get_port_type() != NetNet::PIMPLICIT) {
rc = cur->set_port_type((*decl)[idx]->get_port_type());
assert(rc);
}
if ((*decl)[idx]->get_wire_type() != NetNet::IMPLICIT) {
rc = cur->set_wire_type((*decl)[idx]->get_wire_type());
assert(rc);
}
} else {
defs[pname] = (*decl)[idx];
}
}
/* Put the parameters into a vector of wire descriptions. Look
in the map for the definitions of the name. */
1999-06-15 05:44:53 +02:00
svector<PWire*> pins (parms->size());
{ list<string>::iterator cur;
unsigned idx;
for (cur = parms->begin(), idx = 0
; cur != parms->end()
; idx++, cur++) {
pins[idx] = defs[*cur];
}
}
/* Check that the output is an output and the inputs are
inputs. I can also make sure that only the single output is
declared a register, if anything. */
1999-06-15 05:44:53 +02:00
assert(pins.count() > 0);
assert(pins[0]);
assert(pins[0]->get_port_type() == NetNet::POUTPUT);
1999-06-15 05:44:53 +02:00
for (unsigned idx = 1 ; idx < pins.count() ; idx += 1) {
assert(pins[idx]);
assert(pins[idx]->get_port_type() == NetNet::PINPUT);
assert(pins[idx]->get_wire_type() != NetNet::REG);
}
/* Interpret and check the table entry strings, to make sure
they correspond to the inputs, output and output type. Make
up vectors for the fully interpreted result that can be
placed in the PUdp object. */
1999-06-15 05:44:53 +02:00
svector<string> input (table->size());
svector<char> current (table->size());
svector<char> output (table->size());
{ unsigned idx = 0;
for (list<string>::iterator cur = table->begin()
; cur != table->end()
; cur ++, idx += 1) {
string tmp = *cur;
1999-06-15 05:44:53 +02:00
assert(tmp.find(':') == (pins.count() - 1));
1999-06-15 05:44:53 +02:00
input[idx] = tmp.substr(0, pins.count()-1);
tmp = tmp.substr(pins.count()-1);
if (pins[0]->get_wire_type() == NetNet::REG) {
assert(tmp[0] == ':');
assert(tmp.size() == 4);
current[idx] = tmp[1];
tmp = tmp.substr(2);
}
assert(tmp[0] == ':');
assert(tmp.size() == 2);
output[idx] = tmp[1];
}
}
/* Verify the "initial" statement, if present, to be sure that
it only assignes to the output and the output is
registered. Then save the initial value that I get. */
verinum::V init = verinum::Vx;
if (init_expr) {
// XXXX
assert(pins[0]->get_wire_type() == NetNet::REG);
PAssign*pa = dynamic_cast<PAssign*>(init_expr);
assert(pa);
const PEIdent*id = dynamic_cast<const PEIdent*>(pa->lval());
assert(id);
// XXXX
assert(id->name() == pins[0]->name());
const PENumber*np = dynamic_cast<const PENumber*>(pa->rval());
assert(np);
init = np->value()[0];
}
// Put the primitive into the primitives table
1999-07-10 03:03:18 +02:00
if (vl_primitives[name]) {
VLerror("UDP primitive already exists.");
} else {
1999-07-10 03:03:18 +02:00
PUdp*udp = new PUdp(name, parms->size());
// Detect sequential udp.
if (pins[0]->get_wire_type() == NetNet::REG)
udp->sequential = true;
// Make the port list for the UDP
1999-06-15 05:44:53 +02:00
for (unsigned idx = 0 ; idx < pins.count() ; idx += 1)
udp->ports[idx] = pins[idx]->name();
udp->tinput = input;
udp->tcurrent = current;
udp->toutput = output;
udp->initial = init;
1999-07-10 03:03:18 +02:00
vl_primitives[name] = udp;
}
/* Delete the excess tables and lists from the parser. */
delete parms;
delete decl;
delete table;
delete init_expr;
}
2000-04-01 21:31:57 +02:00
/*
* This is invoked to make a named event. This is the declaration of
* the event, and not necessarily the use of it.
*/
static void pform_make_event(const string&name, const string&fn, unsigned ln)
{
PEvent*event = new PEvent(name);
event->set_file(fn);
event->set_lineno(ln);
pform_cur_module->events[name] = event;
}
void pform_make_events(const list<string>*names, const string&fn, unsigned ln)
{
list<string>::const_iterator cur;
for (cur = names->begin() ; cur != names->end() ; cur++)
pform_make_event(*cur, fn, ln);
}
1999-02-15 03:06:15 +01:00
/*
* pform_makegates is called when a list of gates (with the same type)
* are ready to be instantiated. The function runs through the list of
1999-05-29 04:36:17 +02:00
* gates and calls the pform_makegate function to make the individual gate.
1999-02-15 03:06:15 +01:00
*/
1998-11-04 00:28:49 +01:00
void pform_makegate(PGBuiltin::Type type,
struct str_pair_t str,
svector<PExpr*>* delay,
1999-02-15 03:06:15 +01:00
const lgate&info)
1998-11-04 00:28:49 +01:00
{
1999-05-29 04:36:17 +02:00
if (info.parms_by_name) {
cerr << info.file << ":" << info.lineno << ": Gates do not "
"have port names." << endl;
error_count += 1;
return;
}
PGBuiltin*cur = new PGBuiltin(type, info.name, info.parms, delay);
1999-02-15 03:06:15 +01:00
if (info.range[0])
cur->set_range(info.range[0], info.range[1]);
cur->strength0(str.str0);
cur->strength1(str.str1);
1999-02-15 03:06:15 +01:00
cur->set_file(info.file);
cur->set_lineno(info.lineno);
1999-07-03 04:12:51 +02:00
pform_cur_module->add_gate(cur);
1998-11-04 00:28:49 +01:00
}
void pform_makegates(PGBuiltin::Type type,
struct str_pair_t str,
svector<PExpr*>*delay,
svector<lgate>*gates)
1998-11-04 00:28:49 +01:00
{
1999-05-31 17:45:59 +02:00
for (unsigned idx = 0 ; idx < gates->count() ; idx += 1) {
pform_makegate(type, str, delay, (*gates)[idx]);
1998-11-04 00:28:49 +01:00
}
delete gates;
}
1999-05-29 04:36:17 +02:00
/*
* A module is different from a gate in that there are different
2000-02-18 06:15:02 +01:00
* constraints, and sometimes different syntax. The X_modgate
* functions handle the instantaions of modules (and UDP objects) by
* making PGModule objects.
1999-05-29 04:36:17 +02:00
*/
static void pform_make_modgate(const string&type,
const string&name,
struct parmvalue_t*overrides,
1999-05-29 04:36:17 +02:00
svector<PExpr*>*wires,
2000-02-18 06:15:02 +01:00
PExpr*msb, PExpr*lsb,
const string&fn, unsigned ln)
1998-11-04 00:28:49 +01:00
{
1999-05-03 01:25:32 +02:00
if (name == "") {
cerr << fn << ":" << ln << ": Instantiation of " << type
<< " module requires an instance name." << endl;
error_count += 1;
return;
}
PGModule*cur = new PGModule(type, name, wires);
cur->set_file(fn);
cur->set_lineno(ln);
2000-02-18 06:15:02 +01:00
cur->set_range(msb,lsb);
if (overrides && overrides->by_name) {
unsigned cnt = overrides->by_name->count();
named<PExpr*>*byname = new named<PExpr*>[cnt];
for (unsigned idx = 0 ; idx < cnt ; idx += 1) {
portname_t*curp = (*overrides->by_name)[idx];
byname[idx].name = curp->name;
byname[idx].parm = curp->parm;
}
cur->set_parameters(byname, cnt);
} else if (overrides && overrides->by_order) {
cur->set_parameters(overrides->by_order);
}
1999-07-03 04:12:51 +02:00
pform_cur_module->add_gate(cur);
1998-11-04 00:28:49 +01:00
}
1999-05-29 04:36:17 +02:00
static void pform_make_modgate(const string&type,
const string&name,
struct parmvalue_t*overrides,
1999-05-29 04:36:17 +02:00
svector<portname_t*>*bind,
2000-02-18 06:15:02 +01:00
PExpr*msb, PExpr*lsb,
1999-05-29 04:36:17 +02:00
const string&fn, unsigned ln)
{
if (name == "") {
cerr << fn << ":" << ln << ": Instantiation of " << type
<< " module requires an instance name." << endl;
error_count += 1;
return;
}
unsigned npins = bind->count();
named<PExpr*>*pins = new named<PExpr*>[npins];
1999-05-29 04:36:17 +02:00
for (unsigned idx = 0 ; idx < npins ; idx += 1) {
portname_t*curp = (*bind)[idx];
pins[idx].name = curp->name;
pins[idx].parm = curp->parm;
}
PGModule*cur = new PGModule(type, name, pins, npins);
1999-05-29 04:36:17 +02:00
cur->set_file(fn);
cur->set_lineno(ln);
2000-02-18 06:15:02 +01:00
cur->set_range(msb,lsb);
if (overrides && overrides->by_name) {
unsigned cnt = overrides->by_name->count();
named<PExpr*>*byname = new named<PExpr*>[cnt];
for (unsigned idx = 0 ; idx < cnt ; idx += 1) {
portname_t*curp = (*overrides->by_name)[idx];
byname[idx].name = curp->name;
byname[idx].parm = curp->parm;
}
cur->set_parameters(byname, cnt);
} else if (overrides && overrides->by_order) {
cur->set_parameters(overrides->by_order);
}
1999-07-03 04:12:51 +02:00
pform_cur_module->add_gate(cur);
1999-05-29 04:36:17 +02:00
}
void pform_make_modgates(const string&type,
struct parmvalue_t*overrides,
svector<lgate>*gates)
1998-11-04 00:28:49 +01:00
{
if (overrides && overrides->by_order)
for (unsigned idx = 0 ; idx < overrides->by_order->count() ; idx += 1)
if (! pform_expression_is_constant((*overrides->by_order)[idx])) {
2000-01-02 02:59:28 +01:00
VLerror("error: Parameter override expression"
" must be constant.");
return;
}
2000-01-02 00:47:58 +01:00
1999-05-06 06:37:17 +02:00
for (unsigned idx = 0 ; idx < gates->count() ; idx += 1) {
lgate cur = (*gates)[idx];
1998-11-04 00:28:49 +01:00
1999-05-29 04:36:17 +02:00
if (cur.parms_by_name) {
2000-02-18 06:15:02 +01:00
pform_make_modgate(type, cur.name, overrides,
cur.parms_by_name,
cur.range[0], cur.range[1],
1999-05-29 04:36:17 +02:00
cur.file, cur.lineno);
} else if (cur.parms) {
/* If there are no parameters, the parser will be
tricked into thinking it is one empty
parameter. This fixes that. */
if ((cur.parms->count() == 1) && (cur.parms[0][0] == 0)) {
delete cur.parms;
cur.parms = new svector<PExpr*>(0);
}
2000-02-18 06:15:02 +01:00
pform_make_modgate(type, cur.name, overrides,
cur.parms,
cur.range[0], cur.range[1],
cur.file, cur.lineno);
} else {
1999-05-29 04:36:17 +02:00
svector<PExpr*>*wires = new svector<PExpr*>(0);
2000-02-18 06:15:02 +01:00
pform_make_modgate(type, cur.name, overrides,
wires,
cur.range[0], cur.range[1],
cur.file, cur.lineno);
1998-11-04 00:28:49 +01:00
}
}
delete gates;
}
PGAssign* pform_make_pgassign(PExpr*lval, PExpr*rval,
2000-05-06 17:41:56 +02:00
svector<PExpr*>*del,
struct str_pair_t str)
1998-11-04 00:28:49 +01:00
{
1999-05-29 04:36:17 +02:00
svector<PExpr*>*wires = new svector<PExpr*>(2);
(*wires)[0] = lval;
(*wires)[1] = rval;
PGAssign*cur;
if (del == 0)
cur = new PGAssign(wires);
else
cur = new PGAssign(wires, del);
2000-05-06 17:41:56 +02:00
cur->strength0(str.str0);
cur->strength1(str.str1);
1999-07-03 04:12:51 +02:00
pform_cur_module->add_gate(cur);
return cur;
1998-11-04 00:28:49 +01:00
}
1999-08-27 17:08:37 +02:00
void pform_make_pgassign_list(svector<PExpr*>*alist,
svector<PExpr*>*del,
2000-05-06 17:41:56 +02:00
struct str_pair_t str,
1999-08-27 17:08:37 +02:00
const string& text,
unsigned lineno)
{
PGAssign*tmp;
for (unsigned idx = 0 ; idx < alist->count()/2 ; idx += 1) {
tmp = pform_make_pgassign((*alist)[2*idx],
2000-05-06 17:41:56 +02:00
(*alist)[2*idx+1],
del, str);
1999-08-27 17:08:37 +02:00
tmp->set_file(text);
tmp->set_lineno(lineno);
}
}
1999-12-30 20:06:14 +01:00
/*
* this function makes the initial assignment to a register as given
* in the source. It handles the case where a reg variable is assigned
* where it it declared:
*
* reg foo = <expr>;
*
* This is equivilent to the combination of statements:
*
* reg foo;
* initital foo = <expr>;
*
* and that is how it is parsed. This syntax is not part of the
* IEEE1364-1994 standard, but is approved by OVI as enhancement
* BTF-B14.
*/
void pform_make_reginit(const struct vlltype&li,
const string&name, PExpr*expr)
{
const string sname = scoped_name(name);
PWire*cur = pform_cur_module->get_wire(sname);
if (cur == 0) {
VLerror(li, "internal error: reginit to non-register?");
delete expr;
return;
}
PEIdent*lval = new PEIdent(sname);
lval->set_file(li.text);
lval->set_lineno(li.first_line);
PAssign*ass = new PAssign(lval, expr);
ass->set_file(li.text);
ass->set_lineno(li.first_line);
PProcess*top = new PProcess(PProcess::PR_INITIAL, ass);
top->set_file(li.text);
top->set_lineno(li.first_line);
pform_cur_module->add_behavior(top);
}
void pform_makewire(const vlltype&li, const string&nm,
1999-06-02 17:38:46 +02:00
NetNet::Type type)
1998-11-04 00:28:49 +01:00
{
const string name = scoped_name(nm);
1999-07-03 04:12:51 +02:00
PWire*cur = pform_cur_module->get_wire(name);
1998-11-04 00:28:49 +01:00
if (cur) {
if (cur->get_wire_type() != NetNet::IMPLICIT) {
strstream msg;
1999-06-02 17:38:46 +02:00
msg << name << " previously defined at " <<
1999-12-30 20:06:14 +01:00
cur->get_line() << "." << ends;
VLerror(msg.str());
} else {
bool rc = cur->set_wire_type(type);
assert(rc);
}
1998-11-04 00:28:49 +01:00
return;
}
cur = new PWire(name, type, NetNet::NOT_A_PORT);
1999-06-02 17:38:46 +02:00
cur->set_file(li.text);
cur->set_lineno(li.first_line);
1999-07-03 04:12:51 +02:00
pform_cur_module->add_wire(cur);
1998-11-04 00:28:49 +01:00
}
1999-06-02 17:38:46 +02:00
void pform_makewire(const vlltype&li, const list<string>*names,
NetNet::Type type)
1998-11-04 00:28:49 +01:00
{
for (list<string>::const_iterator cur = names->begin()
; cur != names->end()
; cur ++ )
1999-06-02 17:38:46 +02:00
pform_makewire(li, *cur, type);
1998-11-04 00:28:49 +01:00
}
void pform_set_port_type(const string&nm, NetNet::PortType pt)
1998-11-04 00:28:49 +01:00
{
const string name = scoped_name(nm);
1999-07-03 04:12:51 +02:00
PWire*cur = pform_cur_module->get_wire(name);
1998-11-04 00:28:49 +01:00
if (cur == 0) {
cur = new PWire(name, NetNet::IMPLICIT, pt);
pform_cur_module->add_wire(cur);
1998-11-04 00:28:49 +01:00
}
if (! cur->set_port_type(pt))
1998-11-04 00:28:49 +01:00
VLerror("error setting port direction.");
}
1999-07-24 04:11:19 +02:00
/*
* This function is called by the parser to create task ports. The
* resulting wire (which should be a register) is put into a list to
* be packed into the task parameter list.
*
* It is possible that the wire (er, register) was already created,
* but we know that if the name matches it is a part of the current
* task, so in that case I just assign direction to it.
*
* The following example demonstrates some of the issues:
1999-07-24 04:11:19 +02:00
*
* task foo;
* input a;
* reg a, b;
* input b;
* [...]
* endtask
*
* This function is called when the parser matches the "input a" and
* the "input b" statements. For ``a'', this function is called before
* the wire is declared as a register, so I create the foo.a
* wire. For ``b'', I will find that there is already a foo.b and I
* just set the port direction. In either case, the ``reg a, b''
* statement is caught by the block_item non-terminal and processed
* there.
*
* Ports are implicitly type reg, because it must be possible for the
* port to act as an l-value in a procedural assignment. It is obvious
* for output and inout ports that the type is reg, because the task
* only contains behavior (no structure) to a procedural assignment is
1999-07-31 21:15:21 +02:00
* the *only* way to affect the output. It is less obvious for input
* ports, but in practice an input port receives its value as if by a
* procedural assignment from the calling behavior.
*
* This function also handles the input ports of function
* definitions. Input ports to function definitions have the same
* constraints as those of tasks, so this works fine. Functions have
* no output or inout ports.
1999-07-24 04:11:19 +02:00
*/
svector<PWire*>*pform_make_task_ports(NetNet::PortType pt,
const svector<PExpr*>*range,
1999-09-10 07:02:09 +02:00
const list<string>*names,
const string& file,
unsigned lineno)
1999-07-24 04:11:19 +02:00
{
1999-09-10 07:02:09 +02:00
assert(names);
1999-07-24 04:11:19 +02:00
svector<PWire*>*res = new svector<PWire*>(0);
for (list<string>::const_iterator cur = names->begin()
; cur != names->end() ; cur ++ ) {
string name = scoped_name(*cur);
/* Look for a preexisting wire. If it exists, set the
port direction. If not, create it. */
PWire*curw = pform_cur_module->get_wire(name);
if (curw) {
curw->set_port_type(pt);
} else {
curw = new PWire(name, NetNet::IMPLICIT_REG, pt);
1999-09-10 07:02:09 +02:00
curw->set_file(file);
curw->set_lineno(lineno);
1999-07-24 04:11:19 +02:00
pform_cur_module->add_wire(curw);
}
/* If there is a range involved, it needs to be set. */
if (range)
curw->set_range((*range)[0], (*range)[1]);
svector<PWire*>*tmp = new svector<PWire*>(*res, curw);
delete res;
res = tmp;
}
return res;
}
1999-07-03 04:12:51 +02:00
void pform_set_task(const string&name, PTask*task)
{
pform_cur_module->add_task(name, task);
}
/*
* This function is called to fill out the definition of the function
* with the trappings that are discovered after the basic function
* name is parsed.
*/
1999-08-26 00:22:41 +02:00
void pform_set_function(const string&name, svector<PExpr*>*ra, PFunction *func)
{
1999-08-26 00:22:41 +02:00
PWire*out = new PWire(name+"."+name, NetNet::REG, NetNet::POUTPUT);
if (ra) {
assert(ra->count() == 2);
out->set_range((*ra)[0], (*ra)[1]);
delete ra;
}
pform_cur_module->add_wire(out);
func->set_output(out);
pform_cur_module->add_function(name, func);
}
void pform_set_attrib(const string&name, const string&key, const string&value)
{
1999-07-03 04:12:51 +02:00
PWire*cur = pform_cur_module->get_wire(name);
if (PWire*cur = pform_cur_module->get_wire(name)) {
cur->attributes[key] = value;
} else if (PGate*cur = pform_cur_module->get_gate(name)) {
cur->attributes[key] = value;
} else {
VLerror("Unable to match name for setting attribute.");
}
}
/*
* Set the attribute of a TYPE. This is different from an object in
* that this applies to every instantiation of the given type.
*/
void pform_set_type_attrib(const string&name, const string&key,
const string&value)
{
map<string,PUdp*>::const_iterator udp = vl_primitives.find(name);
if (udp == vl_primitives.end()) {
VLerror("type name is not (yet) defined.");
return;
}
(*udp).second ->attributes[key] = value;
}
/*
* This function attaches a memory index range to an existing
* register. (The named wire must be a register.
*/
void pform_set_reg_idx(const string&name, PExpr*l, PExpr*r)
{
1999-07-03 04:12:51 +02:00
PWire*cur = pform_cur_module->get_wire(name);
if (cur == 0) {
VLerror(" error: name is not a valid net.");
return;
}
cur->set_memory_idx(l, r);
}
/*
* This function attaches a range to a given name. The function is
* only called by the parser within the scope of the net declaration,
* and the name that I receive only has the tail component.
*/
static void pform_set_net_range(const string&name, const svector<PExpr*>*range)
1998-11-04 00:28:49 +01:00
{
1999-06-12 05:42:17 +02:00
assert(range);
assert(range->count() == 2);
1998-11-04 00:28:49 +01:00
PWire*cur = pform_cur_module->get_wire(scoped_name(name));
1998-11-04 00:28:49 +01:00
if (cur == 0) {
VLerror(" error: name is not a valid net.");
1998-11-04 00:28:49 +01:00
return;
}
assert((*range)[0]);
assert((*range)[1]);
cur->set_range((*range)[0], (*range)[1]);
1998-11-04 00:28:49 +01:00
}
void pform_set_net_range(list<string>*names, const svector<PExpr*>*range)
{
assert(range->count() == 2);
for (list<string>::const_iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
pform_set_net_range(*cur, range);
}
}
1999-02-21 18:01:57 +01:00
void pform_set_parameter(const string&name, PExpr*expr)
{
assert(expr);
1999-07-03 04:12:51 +02:00
pform_cur_module->parameters[name] = expr;
pform_cur_module->param_names.push_back(name);
1999-02-21 18:01:57 +01:00
}
2000-03-12 18:09:40 +01:00
void pform_set_localparam(const string&name, PExpr*expr)
{
assert(expr);
pform_cur_module->localparams[name] = expr;
}
void pform_set_defparam(const string&name, PExpr*expr)
{
assert(expr);
pform_cur_module->defparms[name] = expr;
}
1998-11-04 00:28:49 +01:00
void pform_set_port_type(list<string>*names, NetNet::PortType pt)
{
for (list<string>::const_iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
pform_set_port_type(*cur, pt);
}
}
1999-09-10 07:02:09 +02:00
static void pform_set_reg_integer(const string&nm)
1998-11-04 00:28:49 +01:00
{
1999-09-10 07:02:09 +02:00
string name = scoped_name(nm);
1999-07-03 04:12:51 +02:00
PWire*cur = pform_cur_module->get_wire(name);
if (cur == 0) {
cur = new PWire(name, NetNet::INTEGER, NetNet::NOT_A_PORT);
pform_cur_module->add_wire(cur);
} else {
bool rc = cur->set_wire_type(NetNet::INTEGER);
assert(rc);
}
assert(cur);
1998-11-04 00:28:49 +01:00
cur->set_range(new PENumber(new verinum(INTEGER_WIDTH-1, INTEGER_WIDTH)),
new PENumber(new verinum(0UL, INTEGER_WIDTH)));
}
void pform_set_reg_integer(list<string>*names)
{
1998-11-04 00:28:49 +01:00
for (list<string>::const_iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
pform_set_reg_integer(*cur);
1998-11-04 00:28:49 +01:00
}
}
1999-06-12 22:35:27 +02:00
svector<PWire*>* pform_make_udp_input_ports(list<string>*names)
{
1999-06-12 22:35:27 +02:00
svector<PWire*>*out = new svector<PWire*>(names->size());
1999-06-12 22:35:27 +02:00
unsigned idx = 0;
for (list<string>::const_iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
PWire*pp = new PWire(*cur, NetNet::IMPLICIT, NetNet::PINPUT);
1999-06-12 22:35:27 +02:00
(*out)[idx] = pp;
1999-06-21 03:02:16 +02:00
idx += 1;
}
delete names;
return out;
}
PProcess* pform_make_behavior(PProcess::Type type, Statement*st)
1998-11-04 00:28:49 +01:00
{
PProcess*pp = new PProcess(type, st);
1999-07-03 04:12:51 +02:00
pform_cur_module->add_behavior(pp);
return pp;
1998-11-04 00:28:49 +01:00
}
FILE*vl_input = 0;
1998-12-09 05:02:47 +01:00
int pform_parse(const char*path, map<string,Module*>&modules,
map<string,PUdp*>&prim)
1998-11-04 00:28:49 +01:00
{
1998-12-09 05:02:47 +01:00
vl_file = path;
1999-11-23 02:04:57 +01:00
if (strcmp(path, "-") == 0)
vl_input = stdin;
else
vl_input = fopen(path, "r");
1998-12-09 05:02:47 +01:00
if (vl_input == 0) {
cerr << "Unable to open " <<vl_file << "." << endl;
return 11;
}
error_count = 0;
warn_count = 0;
1998-11-04 00:28:49 +01:00
int rc = VLparse();
if (rc) {
cerr << "I give up." << endl;
}
modules = vl_modules;
prim = vl_primitives;
return error_count;
1998-11-04 00:28:49 +01:00
}
/*
* $Log: pform.cc,v $
2000-07-29 19:58:20 +02:00
* Revision 1.63 2000/07/29 17:58:21 steve
* Introduce min:typ:max support.
*
* Revision 1.62 2000/07/22 22:09:04 steve
* Parse and elaborate timescale to scopes.
*
* Revision 1.61 2000/05/23 16:03:13 steve
* Better parsing of expressions lists will empty expressoins.
*
* Revision 1.60 2000/05/16 04:05:16 steve
* Module ports are really special PEIdent
* expressions, because a name can be used
* many places in the port list.
*
* Revision 1.59 2000/05/08 05:30:20 steve
* Deliver gate output strengths to the netlist.
*
2000-05-06 17:41:56 +02:00
* Revision 1.58 2000/05/06 15:41:57 steve
* Carry assignment strength to pform.
*
2000-04-01 21:31:57 +02:00
* Revision 1.57 2000/04/01 19:31:57 steve
* Named events as far as the pform.
*
2000-03-12 18:09:40 +01:00
* Revision 1.56 2000/03/12 17:09:41 steve
* Support localparam.
*
* Revision 1.55 2000/03/08 04:36:54 steve
* Redesign the implementation of scopes and parameters.
* I now generate the scopes and notice the parameters
* in a separate pass over the pform. Once the scopes
* are generated, I can process overrides and evalutate
* paremeters before elaboration begins.
*
* Revision 1.54 2000/02/23 02:56:55 steve
* Macintosh compilers do not support ident.
*
2000-02-18 06:15:02 +01:00
* Revision 1.53 2000/02/18 05:15:03 steve
* Catch module instantiation arrays.
*
* Revision 1.52 2000/01/09 05:50:49 steve
* Support named parameter override lists.
*
2000-01-02 02:59:28 +01:00
* Revision 1.51 2000/01/02 01:59:28 steve
* Forgot to handle no overrides at all.
*
2000-01-02 00:47:58 +01:00
* Revision 1.50 2000/01/01 23:47:58 steve
* Fix module parameter override syntax.
*
1999-12-30 20:06:14 +01:00
* Revision 1.49 1999/12/30 19:06:14 steve
* Support reg initial assignment syntax.
*
* Revision 1.48 1999/12/11 05:45:41 steve
* Fix support for attaching attributes to primitive gates.
*
1999-11-23 02:04:57 +01:00
* Revision 1.47 1999/11/23 01:04:57 steve
* A file name of - means standard input.
*
* Revision 1.46 1999/09/30 01:22:37 steve
* Handle declaration of integers (including scope) in functions.
*
* Revision 1.45 1999/09/21 00:58:33 steve
* Get scope right when setting the net range.
*
1999-09-17 04:06:25 +02:00
* Revision 1.44 1999/09/17 02:06:26 steve
* Handle unconnected module ports.
*
* Revision 1.43 1999/09/15 01:55:06 steve
* Elaborate non-blocking assignment to memories.
*
1999-09-10 07:02:09 +02:00
* Revision 1.42 1999/09/10 05:02:09 steve
* Handle integers at task parameters.
*
* Revision 1.41 1999/08/31 22:38:29 steve
* Elaborate and emit to vvm procedural functions.
*
1999-08-27 17:08:37 +02:00
* Revision 1.40 1999/08/27 15:08:37 steve
* continuous assignment lists.
*
1999-08-26 00:22:41 +02:00
* Revision 1.39 1999/08/25 22:22:41 steve
* elaborate some aspects of functions.
*
* Revision 1.38 1999/08/23 16:48:39 steve
* Parameter overrides support from Peter Monta
* AND and XOR support wide expressions.
*
* Revision 1.37 1999/08/03 04:14:49 steve
* Parse into pform arbitrarily complex module
* port declarations.
1998-11-04 00:28:49 +01:00
*/