iverilog/pform.cc

1162 lines
31 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)
#ident "$Id: pform.cc,v 1.83 2001/10/31 03:11:15 steve Exp $"
1998-11-04 00:28:49 +01:00
#endif
# include "config.h"
# include "compiler.h"
1998-11-04 00:28:49 +01:00
# include "pform.h"
# include "parse_misc.h"
2001-10-21 01:02:39 +02:00
# include "parse_api.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
2001-10-21 01:02:39 +02:00
map<string,Module*> pform_modules;
map<string,PUdp*> pform_primitives;
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;
}
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 char*name, svector<Module::port_t*>*ports,
const char*file, unsigned lineno)
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;
pform_cur_module->set_file(file);
pform_cur_module->set_lineno(lineno);
delete ports;
1998-11-04 00:28:49 +01:00
}
void pform_endmodule(const char*name)
1998-11-04 00:28:49 +01:00
{
1999-07-03 04:12:51 +02:00
assert(pform_cur_module);
assert(strcmp(name, pform_cur_module->mod_name()) == 0);
2001-10-21 01:02:39 +02:00
pform_modules[name] = pform_cur_module;
1999-07-03 04:12:51 +02:00
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 typ;
2000-07-29 19:58:20 +02:00
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,
const char*file, unsigned lineno)
{
unsigned local_errors = 0;
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. In this loop,
the parms list in the list of ports in the port list of the
UDP declaration, and the defs map maps that name to a
PWire* created by an input or output declaration. */
1999-06-15 05:44:53 +02:00
svector<PWire*> pins (parms->size());
svector<string> pin_names (parms->size());
{ list<string>::iterator cur;
unsigned idx;
for (cur = parms->begin(), idx = 0
; cur != parms->end()
; idx++, cur++) {
pins[idx] = defs[*cur];
pin_names[idx] = *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. The possible errors are:
-- an input port (not the first) is missing an input
declaration.
-- An input port is declared output.
*/
1999-06-15 05:44:53 +02:00
assert(pins.count() > 0);
do {
if (pins[0] == 0) {
cerr << file<<":"<<lineno << ": error: "
<< "Output port of primitive " << name
<< " missing output declaration." << endl;
cerr << file<<":"<<lineno << ": : "
<< "Try: output " << pin_names[0] << ";"
<< endl;
error_count += 1;
local_errors += 1;
break;
}
if (pins[0]->get_port_type() != NetNet::POUTPUT) {
cerr << file<<":"<<lineno << ": error: "
<< "The first port of a primitive"
<< " must be an output." << endl;
cerr << file<<":"<<lineno << ": : "
<< "Try: output " << pin_names[0] << ";"
<< endl;
error_count += 1;
local_errors += 1;
break;;
}
} while (0);
1999-06-15 05:44:53 +02:00
for (unsigned idx = 1 ; idx < pins.count() ; idx += 1) {
if (pins[idx] == 0) {
cerr << file<<":"<<lineno << ": error: "
<< "Port " << (idx+1)
<< " of primitive " << name << " missing"
<< " input declaration." << endl;
cerr << file<<":"<<lineno << ": : "
<< "Try: input " << pin_names[idx] << ";"
<< endl;
error_count += 1;
local_errors += 1;
continue;
}
if (pins[idx]->get_port_type() != NetNet::PINPUT) {
cerr << file<<":"<<lineno << ": error: "
<< "Port " << (idx+1)
<< " of primitive " << name << " in an input port"
<< " with an output declaration." << endl;
cerr << file<<":"<<lineno << ": : "
<< "Try: input " << pin_names[idx] << ";"
<< endl;
error_count += 1;
local_errors += 1;
continue;
}
assert(pins[idx]->get_wire_type() != NetNet::REG);
}
if (local_errors > 0) {
delete parms;
delete decl;
delete table;
delete init_expr;
return;
}
/* 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
2001-10-21 01:02:39 +02:00
if (pform_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;
2001-10-21 01:02:39 +02:00
pform_primitives[name] = udp;
}
/* Delete the excess tables and lists from the parser. */
delete parms;
delete decl;
delete table;
delete init_expr;
}
2000-10-31 18:00:04 +01:00
/*
* 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 char*name,
const svector<PExpr*>*range,
bool signed_flag)
2000-10-31 18:00:04 +01:00
{
assert(range);
assert(range->count() == 2);
PWire*cur = pform_cur_module->get_wire(scoped_name(name));
if (cur == 0) {
VLerror("error: name is not a valid net.");
2000-10-31 18:00:04 +01:00
return;
}
assert((*range)[0]);
assert((*range)[1]);
cur->set_range((*range)[0], (*range)[1]);
cur->set_signed(signed_flag);
2000-10-31 18:00:04 +01:00
}
void pform_set_net_range(list<char*>*names,
svector<PExpr*>*range,
bool signed_flag)
2000-10-31 18:00:04 +01:00
{
assert(range->count() == 2);
for (list<char*>::iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
char*txt = *cur;
pform_set_net_range(txt, range, signed_flag);
2000-10-31 18:00:04 +01:00
free(txt);
}
delete names;
delete range;
}
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 char*name, const char*fn, unsigned ln)
2000-04-01 21:31:57 +02:00
{
PEvent*event = new PEvent(name);
event->set_file(fn);
event->set_lineno(ln);
pform_cur_module->events[name] = event;
}
void pform_make_events(list<char*>*names, const char*fn, unsigned ln)
2000-04-01 21:31:57 +02:00
{
2000-10-31 18:00:04 +01:00
list<char*>::iterator cur;
for (cur = names->begin() ; cur != names->end() ; cur++) {
char*txt = *cur;
pform_make_event(txt, fn, ln);
free(txt);
}
delete names;
2000-04-01 21:31:57 +02:00
}
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 char*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 char*fn, unsigned ln)
1998-11-04 00:28:49 +01:00
{
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
}
static void pform_make_modgate(const char*type,
1999-05-29 04:36:17 +02:00
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,
const char*fn, unsigned ln)
1999-05-29 04:36:17 +02:00
{
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 char*type,
struct parmvalue_t*overrides,
svector<lgate>*gates)
1998-11-04 00:28:49 +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,
const char* fn,
1999-08-27 17:08:37 +02:00
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);
tmp->set_file(fn);
1999-08-27 17:08:37 +02:00
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);
}
/*
* This function makes a single signal (a wire, a reg, etc) as
* requested by the parser. The name is unscoped, so I attach the
* current scope to it (with the scoped_name function) and I try to
* resolve it with an existing PWire in the scope.
*
* The wire might already exist because of an implicit declaration in
* a module port, i.e.:
*
* module foo (bar...
*
* reg bar;
*
* The output (or other port direction indicator) may or may not have
* been seen already, so I do not do any cheching with it yet. But I
* do check to see if the name has already been declared, as this
* function is called for every declaration.
*/
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)
&& (cur->get_wire_type() != NetNet::IMPLICIT_REG)) {
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);
if (rc == false) {
strstream msg;
msg << name << " definition conflicts with "
<< "definition at " << cur->get_line()
<< "." << ends;
VLerror(msg.str());
}
}
cur->set_file(li.text);
cur->set_lineno(li.first_line);
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
}
2000-10-31 18:00:04 +01:00
void pform_makewire(const vlltype&li,
svector<PExpr*>*range,
list<char*>*names,
1999-06-02 17:38:46 +02:00
NetNet::Type type)
1998-11-04 00:28:49 +01:00
{
2000-10-31 18:00:04 +01:00
for (list<char*>::iterator cur = names->begin()
1998-11-04 00:28:49 +01:00
; cur != names->end()
2000-10-31 18:00:04 +01:00
; cur ++ ) {
char*txt = *cur;
pform_makewire(li, txt, type);
if (range)
pform_set_net_range(txt, range, false);
2000-10-31 18:00:04 +01:00
free(txt);
}
1998-11-04 00:28:49 +01:00
2000-10-31 18:00:04 +01:00
delete names;
if (range)
delete range;
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,
2000-10-31 18:00:04 +01:00
svector<PExpr*>*range,
list<char*>*names,
const char* file,
1999-09-10 07:02:09 +02:00
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);
2000-10-31 18:00:04 +01:00
for (list<char*>::iterator cur = names->begin()
1999-07-24 04:11:19 +02:00
; cur != names->end() ; cur ++ ) {
2000-10-31 18:00:04 +01:00
char*txt = *cur;
string name = scoped_name(txt);
1999-07-24 04:11:19 +02:00
/* 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);
2000-10-31 18:00:04 +01:00
free(txt);
1999-07-24 04:11:19 +02:00
delete res;
res = tmp;
}
2000-10-31 18:00:04 +01:00
if (range)
delete range;
delete names;
1999-07-24 04:11:19 +02:00
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)
{
2001-10-21 01:02:39 +02:00
map<string,PUdp*>::const_iterator udp = pform_primitives.find(name);
if (udp == pform_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)
{
PWire*cur = pform_cur_module->get_wire(scoped_name(name));
if (cur == 0) {
VLerror("internal error: name is not a valid memory for index.");
return;
}
cur->set_memory_idx(l, r);
}
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;
}
2000-10-31 18:00:04 +01:00
void pform_set_port_type(list<char*>*names,
svector<PExpr*>*range,
NetNet::PortType pt)
1998-11-04 00:28:49 +01:00
{
2000-10-31 18:00:04 +01:00
for (list<char*>::iterator cur = names->begin()
1998-11-04 00:28:49 +01:00
; cur != names->end()
; cur ++ ) {
2000-10-31 18:00:04 +01:00
char*txt = *cur;
pform_set_port_type(txt, pt);
if (range)
pform_set_net_range(txt, range, false);
2000-10-31 18:00:04 +01:00
free(txt);
1998-11-04 00:28:49 +01:00
}
2000-10-31 18:00:04 +01:00
delete names;
if (range)
delete range;
1998-11-04 00:28:49 +01:00
}
2000-10-31 18:00:04 +01:00
static void pform_set_reg_integer(const char*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) {
2001-01-06 03:29:35 +01:00
cur = new PWire(name, NetNet::REG, NetNet::NOT_A_PORT);
cur->set_signed(true);
pform_cur_module->add_wire(cur);
} else {
2001-01-06 03:29:35 +01:00
bool rc = cur->set_wire_type(NetNet::REG);
assert(rc);
2001-01-06 03:29:35 +01:00
cur->set_signed(true);
}
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)));
cur->set_signed(true);
}
2000-10-31 18:00:04 +01:00
void pform_set_reg_integer(list<char*>*names)
{
2000-10-31 18:00:04 +01:00
for (list<char*>::iterator cur = names->begin()
1998-11-04 00:28:49 +01:00
; cur != names->end()
; cur ++ ) {
2000-10-31 18:00:04 +01:00
char*txt = *cur;
pform_set_reg_integer(txt);
free(txt);
1998-11-04 00:28:49 +01:00
}
2000-10-31 18:00:04 +01:00
delete names;
1998-11-04 00:28:49 +01:00
}
2000-10-31 18:49:02 +01:00
static void pform_set_reg_time(const char*nm)
{
string name = scoped_name(nm);
PWire*cur = pform_cur_module->get_wire(name);
if (cur == 0) {
cur = new PWire(name, NetNet::REG, NetNet::NOT_A_PORT);
2000-10-31 18:49:02 +01:00
pform_cur_module->add_wire(cur);
} else {
bool rc = cur->set_wire_type(NetNet::REG);
2000-10-31 18:49:02 +01:00
assert(rc);
}
assert(cur);
cur->set_range(new PENumber(new verinum(TIME_WIDTH-1, INTEGER_WIDTH)),
new PENumber(new verinum(0UL, INTEGER_WIDTH)));
}
void pform_set_reg_time(list<char*>*names)
{
for (list<char*>::iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
char*txt = *cur;
pform_set_reg_time(txt);
free(txt);
}
delete names;
}
2000-10-31 18:00:04 +01:00
svector<PWire*>* pform_make_udp_input_ports(list<char*>*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;
2000-10-31 18:00:04 +01:00
for (list<char*>::iterator cur = names->begin()
; cur != names->end()
; cur ++ ) {
2000-10-31 18:00:04 +01:00
char*txt = *cur;
PWire*pp = new PWire(txt, 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;
2000-10-31 18:00:04 +01:00
free(txt);
}
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;
2001-10-21 01:02:39 +02:00
extern void reset_lexor();
int pform_parse(const char*path, FILE*file)
1998-11-04 00:28:49 +01:00
{
1998-12-09 05:02:47 +01:00
vl_file = path;
2001-10-21 01:02:39 +02:00
if (file == 0) {
if (strcmp(path, "-") == 0)
vl_input = stdin;
else
vl_input = fopen(path, "r");
if (vl_input == 0) {
cerr << "Unable to open " <<vl_file << "." << endl;
return 11;
}
} else {
vl_input = file;
1998-12-09 05:02:47 +01:00
}
2001-10-21 01:02:39 +02:00
reset_lexor();
error_count = 0;
warn_count = 0;
1998-11-04 00:28:49 +01:00
int rc = VLparse();
2001-10-21 01:02:39 +02:00
if (file == 0)
fclose(vl_input);
if (rc) {
cerr << "I give up." << endl;
error_count += 1;
}
return error_count;
1998-11-04 00:28:49 +01:00
}
/*
* $Log: pform.cc,v $
* Revision 1.83 2001/10/31 03:11:15 steve
* detect module ports not declared within the module.
*
* Revision 1.82 2001/10/21 01:55:24 steve
* Error messages for missing UDP port declarations.
*
* Revision 1.81 2001/10/21 00:42:48 steve
* Module types in pform are char* instead of string.
*
2001-10-21 01:02:39 +02:00
* Revision 1.80 2001/10/20 23:02:40 steve
* Add automatic module libraries.
*
* Revision 1.79 2001/10/20 05:21:51 steve
* Scope/module names are char* instead of string.
*
* Revision 1.78 2001/07/25 03:10:49 steve
* Create a config.h.in file to hold all the config
* junk, and support gcc 3.0. (Stephan Boettcher)
*
* Revision 1.77 2001/05/25 02:21:34 steve
* Detect input and input ports declared as reg.
*
* Revision 1.76 2001/05/20 15:03:25 steve
* Deleted wrong time when -Tmax is selected.
*
* Revision 1.75 2001/04/28 23:18:08 steve
* UDP instances need not have user supplied names.
*
* Revision 1.74 2001/02/17 05:15:33 steve
* Allow task ports to be given real types.
*
* Revision 1.73 2001/01/15 00:47:01 steve
* Pass scope type information to the target module.
*
* Revision 1.72 2001/01/14 23:04:56 steve
* Generalize the evaluation of floating point delays, and
* get it working with delay assignment statements.
*
* Allow parameters to be referenced by hierarchical name.
*
* Revision 1.71 2001/01/10 05:32:44 steve
* Match memories within task scopes. (PR#101)
*
* Revision 1.70 2001/01/06 06:31:59 steve
* declaration initialization for time variables.
*
2001-01-06 03:29:35 +01:00
* Revision 1.69 2001/01/06 02:29:36 steve
* Support arrays of integers.
*
* Revision 1.68 2000/12/11 00:31:43 steve
* Add support for signed reg variables,
* simulate in t-vvm signed comparisons.
*
* Revision 1.67 2000/11/30 17:31:42 steve
* Change LineInfo to store const C strings.
1998-11-04 00:28:49 +01:00
*/