Reimplement combinational UDPs.
This commit is contained in:
parent
becda0e26c
commit
b7ef2fcb0a
|
|
@ -17,7 +17,7 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: compile.cc,v 1.195 2005/03/22 05:18:34 steve Exp $"
|
#ident "$Id: compile.cc,v 1.196 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
# include "arith.h"
|
# include "arith.h"
|
||||||
|
|
@ -773,6 +773,26 @@ void inputs_connect(vvp_net_t*fdx, unsigned argc, struct symb_s*argv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void wide_inputs_connect(vvp_wide_fun_core*core,
|
||||||
|
unsigned argc, struct symb_s*argv)
|
||||||
|
{
|
||||||
|
/* Create input functors to receive values from the
|
||||||
|
network. These functors pass the data to the core. */
|
||||||
|
unsigned input_functors = (argc+3) / 4;
|
||||||
|
for (unsigned idx = 0 ; idx < input_functors ; idx += 1) {
|
||||||
|
unsigned base = idx*4;
|
||||||
|
unsigned trans = 4;
|
||||||
|
if (base+trans > argc)
|
||||||
|
trans = argc - base;
|
||||||
|
|
||||||
|
vvp_wide_fun_t*cur = new vvp_wide_fun_t(core, base);
|
||||||
|
vvp_net_t*ptr = new vvp_net_t;
|
||||||
|
ptr->fun = cur;
|
||||||
|
|
||||||
|
inputs_connect(ptr, trans, argv+base);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
struct const_functor_s: public functor_s {
|
struct const_functor_s: public functor_s {
|
||||||
const_functor_s(unsigned str0, unsigned str1)
|
const_functor_s(unsigned str0, unsigned str1)
|
||||||
|
|
@ -1118,13 +1138,9 @@ void compile_force(char*label, struct symb_s signal,
|
||||||
void compile_udp_def(int sequ, char *label, char *name,
|
void compile_udp_def(int sequ, char *label, char *name,
|
||||||
unsigned nin, unsigned init, char **table)
|
unsigned nin, unsigned init, char **table)
|
||||||
{
|
{
|
||||||
struct vvp_udp_s *u = udp_create(label);
|
vvp_udp_s *u = new vvp_udp_s(label, name, nin, sequ? true : false);
|
||||||
u->name = name;
|
u->compile_table(table);
|
||||||
u->sequ = sequ;
|
free(label);
|
||||||
u->nin = nin;
|
|
||||||
u->init = init;
|
|
||||||
u->compile_table(table);
|
|
||||||
free(label);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
char **compile_udp_table(char **table, char *row)
|
char **compile_udp_table(char **table, char *row)
|
||||||
|
|
@ -1143,42 +1159,6 @@ char **compile_udp_table(char **table, char *row)
|
||||||
return table;
|
return table;
|
||||||
}
|
}
|
||||||
|
|
||||||
void compile_udp_functor(char*label, char*type,
|
|
||||||
vvp_delay_t delay,
|
|
||||||
unsigned argc, struct symb_s*argv)
|
|
||||||
{
|
|
||||||
struct vvp_udp_s *u = udp_find(type);
|
|
||||||
assert (argc == u->nin);
|
|
||||||
#if 0
|
|
||||||
functor_t udp = new udp_functor_s(u);
|
|
||||||
|
|
||||||
unsigned nfun = (argc+3)/4;
|
|
||||||
vvp_ipoint_t fdx = functor_allocate(nfun);
|
|
||||||
functor_define(fdx, udp);
|
|
||||||
define_functor_symbol(label, fdx);
|
|
||||||
free(label);
|
|
||||||
|
|
||||||
if (nfun > 1) {
|
|
||||||
for (unsigned i=0; i < nfun-1; i++) {
|
|
||||||
functor_t fu = new edge_inputs_functor_s;
|
|
||||||
vvp_ipoint_t ipt = ipoint_index(fdx, i+1);
|
|
||||||
functor_define(ipt, fu);
|
|
||||||
fu->out = fdx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
udp->delay = delay;
|
|
||||||
|
|
||||||
inputs_connect(fdx, argc, argv);
|
|
||||||
free(argv);
|
|
||||||
|
|
||||||
if (u->sequ)
|
|
||||||
udp->put_oval(u->init, false);
|
|
||||||
#else
|
|
||||||
fprintf(stderr, "XXXX compile_udp_functor not implemented\n");
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Take the detailed parse items from a .mem statement and generate
|
* Take the detailed parse items from a .mem statement and generate
|
||||||
* the necessary internal structures.
|
* the necessary internal structures.
|
||||||
|
|
@ -1587,6 +1567,9 @@ void compile_param_string(char*label, char*name, char*str, char*value)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: compile.cc,v $
|
* $Log: compile.cc,v $
|
||||||
|
* Revision 1.196 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.195 2005/03/22 05:18:34 steve
|
* Revision 1.195 2005/03/22 05:18:34 steve
|
||||||
* The indexed set can write a vector, not just a bit.
|
* The indexed set can write a vector, not just a bit.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
|
|
@ -19,7 +19,7 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: compile.h,v 1.66 2005/03/18 02:56:04 steve Exp $"
|
#ident "$Id: compile.h,v 1.67 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
|
|
@ -49,6 +49,16 @@ extern bool verbose_flag;
|
||||||
extern void inputs_connect(vvp_net_t*fdx, unsigned argc, struct symb_s*argv);
|
extern void inputs_connect(vvp_net_t*fdx, unsigned argc, struct symb_s*argv);
|
||||||
extern void input_connect(vvp_net_t*fdx, unsigned port, char*label);
|
extern void input_connect(vvp_net_t*fdx, unsigned port, char*label);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This function is an expansion of the inputs_connect function. It
|
||||||
|
* uses the inputs_connect function, but it creates vvp_wide_fun_t
|
||||||
|
* nodes to handle arbitrary width nodes, and connects those nodes to
|
||||||
|
* the vvp_wide_fun_core object passed in.
|
||||||
|
*/
|
||||||
|
extern void wide_inputs_connect(vvp_wide_fun_core*core,
|
||||||
|
unsigned argc, struct symb_s*argv);
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Add a functor to the symbol table
|
* Add a functor to the symbol table
|
||||||
*/
|
*/
|
||||||
|
|
@ -300,6 +310,9 @@ extern void compile_net(char*label, char*name,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: compile.h,v $
|
* $Log: compile.h,v $
|
||||||
|
* Revision 1.67 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.66 2005/03/18 02:56:04 steve
|
* Revision 1.66 2005/03/18 02:56:04 steve
|
||||||
* Add support for LPM_UFUNC user defined functions.
|
* Add support for LPM_UFUNC user defined functions.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
604
vvp/udp.cc
604
vvp/udp.cc
|
|
@ -1,6 +1,8 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2000 Stephen Williams (steve@icarus.com)
|
* Copyright (c) 2005 Stephen Williams (steve@icarus.com)
|
||||||
* Copyright (c) 2001 Stephan Boettcher <stephan@nevis.columbia.edu>
|
*
|
||||||
|
* (This is a rewrite of code that was ...
|
||||||
|
* Copyright (c) 2001 Stephan Boettcher <stephan@nevis.columbia.edu>)
|
||||||
*
|
*
|
||||||
* This source code is free software; you can redistribute it
|
* This source code is free software; you can redistribute it
|
||||||
* and/or modify it in source code form under the terms of the GNU
|
* and/or modify it in source code form under the terms of the GNU
|
||||||
|
|
@ -18,445 +20,231 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: udp.cc,v 1.26 2004/10/04 01:10:59 steve Exp $"
|
#ident "$Id: udp.cc,v 1.27 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "udp.h"
|
#include "udp.h"
|
||||||
#include "schedule.h"
|
#include "schedule.h"
|
||||||
#include "symbols.h"
|
#include "symbols.h"
|
||||||
|
#include "compile.h"
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#ifdef HAVE_MALLOC_H
|
#ifdef HAVE_MALLOC_H
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
#endif
|
#endif
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
/*
|
|
||||||
* This method is called when the input of a slice of the UDP
|
|
||||||
* changes. All the slices of the UDP point to this common functor,
|
|
||||||
* that manages the output of the UDP device. The input functors are
|
|
||||||
* all edge_inputs_functors_s objects.
|
|
||||||
*/
|
|
||||||
void udp_functor_s::set(vvp_ipoint_t i, bool push, unsigned val, unsigned)
|
|
||||||
{
|
|
||||||
// Save the input in the ival member of this functor. It will
|
|
||||||
// be read by the propagate method. The old_ival method of the
|
|
||||||
// edge_input_functor (I am that) will be set by propagate.
|
|
||||||
put(i, val);
|
|
||||||
unsigned char out = udp->propagate(this, i);
|
|
||||||
|
|
||||||
// Send the result to the output. If this is a combinational
|
|
||||||
// UDP, then push according to the push flag. However, do
|
|
||||||
// *not* push sequential outputs.
|
|
||||||
|
|
||||||
// Sequential primitive outputs are scheduled as active
|
|
||||||
// events, no matter what common sense and reason say.
|
|
||||||
put_oval(out, push & !udp->sequ, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static symbol_table_t udp_table;
|
static symbol_table_t udp_table;
|
||||||
|
|
||||||
struct vvp_udp_s *udp_create(char *label)
|
struct vvp_udp_s *udp_find(const char *label)
|
||||||
{
|
{
|
||||||
if (!udp_table)
|
symbol_value_t v = sym_get_value(udp_table, label);
|
||||||
udp_table = new_symbol_table();
|
return (struct vvp_udp_s *)v.ptr;
|
||||||
|
|
||||||
assert(!udp_find(label));
|
|
||||||
|
|
||||||
struct vvp_udp_s *u = new vvp_udp_s;
|
|
||||||
|
|
||||||
symbol_value_t v;
|
|
||||||
v.ptr = u;
|
|
||||||
sym_set_value(udp_table, label, v);
|
|
||||||
|
|
||||||
u->name = 0x0;
|
|
||||||
u->sequ = 0;
|
|
||||||
u->nin = 0;
|
|
||||||
u->init = 3;
|
|
||||||
u->table = 0x0;
|
|
||||||
|
|
||||||
return u;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct vvp_udp_s *udp_find(char *label)
|
vvp_udp_s::vvp_udp_s(char*label, char*name, unsigned ports, bool sequ)
|
||||||
{
|
{
|
||||||
symbol_value_t v = sym_get_value(udp_table, label);
|
assert(!sequ); // XXXX sequential UDPs not supported yet.
|
||||||
return (struct vvp_udp_s *)v.ptr;
|
|
||||||
|
if (!udp_table)
|
||||||
|
udp_table = new_symbol_table();
|
||||||
|
|
||||||
|
assert(!udp_find(label));
|
||||||
|
|
||||||
|
symbol_value_t v;
|
||||||
|
v.ptr = this;
|
||||||
|
sym_set_value(udp_table, label, v);
|
||||||
|
|
||||||
|
name_ = name;
|
||||||
|
ports_ = ports;
|
||||||
|
levels0_ = 0;
|
||||||
|
levels1_ = 0;
|
||||||
|
nlevels0_ = 0;
|
||||||
|
nlevels1_ = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef unsigned int udp_vec_t;
|
vvp_udp_s::~vvp_udp_s()
|
||||||
struct udp_table_entry_s
|
|
||||||
{
|
{
|
||||||
udp_vec_t not_0; // all inputs that must not be 0
|
if (levels0_) delete[] levels0_;
|
||||||
udp_vec_t not_1x; // all inputs that must not be 1, x
|
if (levels1_) delete[] levels1_;
|
||||||
unsigned char edge_idx; // input index of the edge
|
}
|
||||||
unsigned char edge_type; // permissible transitions. 0: no edge.
|
|
||||||
unsigned char out; // new output, 0..2
|
|
||||||
};
|
|
||||||
|
|
||||||
enum edge_type_e
|
unsigned vvp_udp_s::port_count() const
|
||||||
{
|
{
|
||||||
EDGE_0 = 0x01,
|
return ports_;
|
||||||
EDGE_1 = 0x02,
|
}
|
||||||
EDGE_x = 0x0c,
|
|
||||||
EDGE_any = 0x0f,
|
vvp_bit4_t vvp_udp_s::test_levels(const udp_levels_table&cur)
|
||||||
};
|
{
|
||||||
|
for (unsigned idx = 0 ; idx < nlevels0_ ; idx += 1) {
|
||||||
|
if (cur.mask0 != levels0_[idx].mask0)
|
||||||
|
continue;
|
||||||
|
if (cur.mask1 != levels0_[idx].mask1)
|
||||||
|
continue;
|
||||||
|
if (cur.maskx != levels0_[idx].maskx)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
return BIT4_0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (unsigned idx = 0 ; idx < nlevels1_ ; idx += 1) {
|
||||||
|
if (cur.mask0 != levels1_[idx].mask0)
|
||||||
|
continue;
|
||||||
|
if (cur.mask1 != levels1_[idx].mask1)
|
||||||
|
continue;
|
||||||
|
if (cur.maskx != levels1_[idx].maskx)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
return BIT4_1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return BIT4_X;
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_udp_s::compile_table(char**tab)
|
||||||
|
{
|
||||||
|
unsigned nrows0 = 0, nrows1 = 0;
|
||||||
|
|
||||||
|
/* First run through the table to figure out the number of
|
||||||
|
rows I need for each kind of table. */
|
||||||
|
for (unsigned idx = 0 ; tab[idx] ; idx += 1) {
|
||||||
|
assert(strlen(tab[idx]) == ports_ + 1);
|
||||||
|
switch (tab[idx][ports_]) {
|
||||||
|
case '0':
|
||||||
|
nrows0 += 1;
|
||||||
|
break;
|
||||||
|
case '1':
|
||||||
|
nrows1 += 1;
|
||||||
|
break;
|
||||||
|
case 'x':
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
nlevels0_ = nrows0;
|
||||||
|
levels0_ = new udp_levels_table[nlevels0_];
|
||||||
|
|
||||||
|
nlevels1_ = nrows1;
|
||||||
|
levels1_ = new udp_levels_table[nlevels1_];
|
||||||
|
|
||||||
|
nrows0 = 0;
|
||||||
|
nrows1 = 0;
|
||||||
|
for (unsigned idx = 0 ; tab[idx] ; idx += 1) {
|
||||||
|
struct udp_levels_table cur;
|
||||||
|
cur.mask0 = 0;
|
||||||
|
cur.mask1 = 0;
|
||||||
|
cur.maskx = 0;
|
||||||
|
assert(ports_ <= sizeof(cur.mask0));
|
||||||
|
for (unsigned pp = 0 ; pp < ports_ ; pp += 1) {
|
||||||
|
unsigned long mask_bit = 1UL << pp;
|
||||||
|
switch (tab[idx][pp]) {
|
||||||
|
case '0':
|
||||||
|
cur.mask0 |= mask_bit;
|
||||||
|
break;
|
||||||
|
case '1':
|
||||||
|
cur.mask1 |= mask_bit;
|
||||||
|
break;
|
||||||
|
case 'x':
|
||||||
|
cur.maskx |= mask_bit;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (tab[idx][ports_]) {
|
||||||
|
case '0':
|
||||||
|
levels0_[nrows0++] = cur;
|
||||||
|
break;
|
||||||
|
case '1':
|
||||||
|
levels1_[nrows1++] = cur;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(nrows0 == nlevels0_);
|
||||||
|
assert(nrows1 == nlevels1_);
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_udp_fun_core::vvp_udp_fun_core(vvp_net_t*net, vvp_udp_s*def)
|
||||||
|
: vvp_wide_fun_core(net, def->port_count())
|
||||||
|
{
|
||||||
|
def_ = def;
|
||||||
|
// Assume initially that all the inputs are 1'bx
|
||||||
|
current_.mask0 = 0;
|
||||||
|
current_.mask1 = 0;
|
||||||
|
current_.maskx = ~ ((-1UL) << port_count());
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_udp_fun_core::~vvp_udp_fun_core()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_udp_fun_core::recv_vec4_from_inputs(unsigned port)
|
||||||
|
{
|
||||||
|
/* For now, assume udps are 1-bit wide. */
|
||||||
|
assert(value(port).size() == 1);
|
||||||
|
|
||||||
|
unsigned long mask = 1UL << port;
|
||||||
|
|
||||||
|
switch (value(port).value(0)) {
|
||||||
|
|
||||||
|
case BIT4_0:
|
||||||
|
current_.mask0 |= mask;
|
||||||
|
current_.mask1 &= ~mask;
|
||||||
|
current_.maskx &= ~mask;
|
||||||
|
break;
|
||||||
|
case BIT4_1:
|
||||||
|
current_.mask0 &= ~mask;
|
||||||
|
current_.mask1 |= mask;
|
||||||
|
current_.maskx &= ~mask;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
current_.mask0 &= ~mask;
|
||||||
|
current_.mask1 &= ~mask;
|
||||||
|
current_.maskx |= mask;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_vector4_t out (1);
|
||||||
|
out.set_bit(0, def_->test_levels(current_));
|
||||||
|
|
||||||
|
propagate_vec4(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This method of the common table object for the UDP calculates the
|
* This function is called by the parser in response to a .udp
|
||||||
* output based on the new input of the functor calling me.
|
* node. We create the nodes needed to integrate the UDP into the
|
||||||
|
* netlist. The definition should be parsed already.
|
||||||
*/
|
*/
|
||||||
unsigned char vvp_udp_s::propagate(functor_t fu, vvp_ipoint_t uix)
|
void compile_udp_functor(char*label, char*type,
|
||||||
|
vvp_delay_t delay,
|
||||||
|
unsigned argc, struct symb_s*argv)
|
||||||
{
|
{
|
||||||
vvp_ipoint_t base = ipoint_make(uix, 0);
|
struct vvp_udp_s *def = udp_find(type);
|
||||||
|
assert(def);
|
||||||
|
free(type);
|
||||||
|
|
||||||
unsigned char ret = 2;
|
vvp_net_t*ptr = new vvp_net_t;
|
||||||
|
vvp_udp_fun_core*core = new vvp_udp_fun_core(ptr, def);
|
||||||
unsigned edge_idx = 0; // input index that changed
|
ptr->fun = core;
|
||||||
unsigned edge_type = 0; // input transition
|
define_functor_symbol(label, ptr);
|
||||||
|
free(label);
|
||||||
udp_vec_t invec = 0x0; // vector of 2-bit inputs
|
|
||||||
|
|
||||||
for (unsigned i=0; i < nin; i+=4)
|
|
||||||
{
|
|
||||||
int idx = ipoint_input_index(base, i);
|
|
||||||
edge_inputs_functor_s *pfun =
|
|
||||||
dynamic_cast<edge_inputs_functor_s *>(functor_index(idx));
|
|
||||||
assert(pfun);
|
|
||||||
|
|
||||||
invec |= pfun->ival << (2*i);
|
|
||||||
|
|
||||||
unsigned char diff = pfun->ival ^ pfun->old_ival;
|
|
||||||
if (diff)
|
|
||||||
{
|
|
||||||
unsigned ii = 0;
|
|
||||||
if (diff & 0x03) ii = 0;
|
|
||||||
if (diff & 0x0c) ii = 1;
|
|
||||||
if (diff & 0x30) ii = 2;
|
|
||||||
if (diff & 0xc0) ii = 3;
|
|
||||||
|
|
||||||
edge_idx = i+ii;
|
|
||||||
|
|
||||||
unsigned old_in = (pfun->old_ival >> (2*ii)) & 3;
|
|
||||||
edge_type = (1<<old_in);
|
|
||||||
}
|
|
||||||
|
|
||||||
pfun->old_ival = pfun->ival;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sequ)
|
|
||||||
{
|
|
||||||
if (edge_type == 0)
|
|
||||||
return fu->get_oval();
|
|
||||||
invec <<= 2;
|
|
||||||
invec |= (fu->get_oval() & 3);
|
|
||||||
}
|
|
||||||
|
|
||||||
udp_vec_t inx = invec & 0xaaaaaaaaU; // all 'x'/'z'
|
|
||||||
udp_vec_t in01 = ~(inx>>1); // masks all 'x'/'z'
|
|
||||||
udp_vec_t in1x = invec & in01; // all 'x' and '1'
|
|
||||||
udp_vec_t in0 = ~invec & in01; // all '0'
|
|
||||||
|
|
||||||
for (unsigned ri=0; ri < ntable; ri++)
|
|
||||||
{
|
|
||||||
udp_table_entry_t row = table+ri;
|
|
||||||
|
|
||||||
if ((in1x & row->not_1x) || (in0 & row->not_0))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (!row->edge_type)
|
|
||||||
{
|
|
||||||
ret = row->out;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (row->edge_idx != edge_idx)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (row->edge_type & edge_type)
|
|
||||||
{
|
|
||||||
ret = row->out;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret>2)
|
|
||||||
ret = fu->get_oval();
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void vvp_udp_s::compile_table(char **tab)
|
|
||||||
{
|
|
||||||
ntable = 0;
|
|
||||||
for (char **ss = tab; *ss; ss++)
|
|
||||||
ntable++;
|
|
||||||
table = new struct udp_table_entry_s[ntable];
|
|
||||||
for (unsigned i = 0; i < ntable; i++)
|
|
||||||
{
|
|
||||||
compile_row_(&table[i], tab[i]);
|
|
||||||
free(tab[i]);
|
|
||||||
}
|
|
||||||
free(tab);
|
|
||||||
}
|
|
||||||
|
|
||||||
void vvp_udp_s::compile_row_(udp_table_entry_t row, char *rchr)
|
|
||||||
{
|
|
||||||
row->not_0 = 0; // all inputs that must not be 0
|
|
||||||
row->not_1x = 0; // all inputs that must not be 1 or x
|
|
||||||
row->edge_idx = 0; // input index of the edge
|
|
||||||
row->edge_type = 0; // permissible transitions. 0: no edge.
|
|
||||||
|
|
||||||
char *s = rchr;
|
|
||||||
for (unsigned i = (sequ ? 0 : 1); i <= nin; i++)
|
|
||||||
{
|
|
||||||
char c = *s;
|
|
||||||
s++;
|
|
||||||
|
|
||||||
unsigned char n0 = 0;
|
|
||||||
unsigned char n1x = 0;
|
|
||||||
unsigned char edge = 0;
|
|
||||||
|
|
||||||
switch (c)
|
|
||||||
{
|
|
||||||
default:
|
|
||||||
fprintf(stderr, "vvp: Illegal character (%d) in UDP table\n", c);
|
|
||||||
assert(0);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '?':
|
|
||||||
break;
|
|
||||||
case '0':
|
|
||||||
n1x = 3; // 1, x not allowed
|
|
||||||
break;
|
|
||||||
case '1':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
break;
|
|
||||||
case 'b':
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
break;
|
|
||||||
case 'l':
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
break;
|
|
||||||
case 'h':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '*':
|
|
||||||
edge = EDGE_any;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '+':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
edge = EDGE_any;
|
|
||||||
break;
|
|
||||||
case '_':
|
|
||||||
n1x = 3; // 1, x not allowed
|
|
||||||
edge = EDGE_any;
|
|
||||||
break;
|
|
||||||
case '%':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
edge = EDGE_any;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'N':
|
|
||||||
edge = EDGE_1;
|
|
||||||
break;
|
|
||||||
case 'P':
|
|
||||||
edge = EDGE_0;
|
|
||||||
break;
|
|
||||||
case 'B':
|
|
||||||
edge = EDGE_x;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'r':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
edge = EDGE_0;
|
|
||||||
break;
|
|
||||||
case 'R':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
edge = EDGE_x;
|
|
||||||
break;
|
|
||||||
case 'f':
|
|
||||||
n1x = 3; // 1, x not allowed
|
|
||||||
edge = EDGE_1;
|
|
||||||
break;
|
|
||||||
case 'F':
|
|
||||||
n1x = 3; // 1, x not allowed
|
|
||||||
edge = EDGE_x;
|
|
||||||
break;
|
|
||||||
case 'Q':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
edge = EDGE_0;
|
|
||||||
break;
|
|
||||||
case 'q':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
edge = EDGE_0 | EDGE_1;
|
|
||||||
break;
|
|
||||||
case 'M':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
edge = EDGE_1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'n':
|
|
||||||
n1x = 1; // 1 not allowed
|
|
||||||
edge = EDGE_1 | EDGE_x;
|
|
||||||
break;
|
|
||||||
case 'p':
|
|
||||||
n0 = 1; // 0 not allowed
|
|
||||||
edge = EDGE_0 | EDGE_x;
|
|
||||||
break;
|
|
||||||
case 'v':
|
|
||||||
n1x = 2; // x not allowed
|
|
||||||
edge = EDGE_0 | EDGE_1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (edge)
|
|
||||||
{
|
|
||||||
if (!sequ)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "vvp: edge in combinatorial UDP\n");
|
|
||||||
assert(0);
|
|
||||||
}
|
|
||||||
if (!i)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "vvp: edge in UDP output state\n");
|
|
||||||
assert(0);
|
|
||||||
}
|
|
||||||
row->edge_idx = i-1;
|
|
||||||
if (row->edge_type)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "vvp: multiple edges in UDP table row\n");
|
|
||||||
assert(0);
|
|
||||||
}
|
|
||||||
row->edge_type = edge;
|
|
||||||
}
|
|
||||||
|
|
||||||
int j = sequ ? i : i-1;
|
|
||||||
row->not_0 |= n0 << (2*j);
|
|
||||||
row->not_1x |= n1x << (2*j);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (*s)
|
|
||||||
{
|
|
||||||
case '0':
|
|
||||||
row->out = 0;
|
|
||||||
break;
|
|
||||||
case '1':
|
|
||||||
row->out = 1;
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
row->out = 2;
|
|
||||||
break;
|
|
||||||
case '-':
|
|
||||||
row->out = 4;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
fprintf(stderr, "vvp: illegal character (%d) in udp output spec\n", *s);
|
|
||||||
assert(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
wide_inputs_connect(core, argc, argv);
|
||||||
|
free(argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: udp.cc,v $
|
* $Log: udp.cc,v $
|
||||||
* Revision 1.26 2004/10/04 01:10:59 steve
|
* Revision 1.27 2005/04/01 06:02:45 steve
|
||||||
* Clean up spurious trailing white space.
|
* Reimplement combinational UDPs.
|
||||||
*
|
|
||||||
* Revision 1.25 2003/09/24 20:46:48 steve
|
|
||||||
* Standard udp scheduling behavior.
|
|
||||||
*
|
|
||||||
* Revision 1.24 2003/09/17 03:39:55 steve
|
|
||||||
* Internal documentation of UDP devices.
|
|
||||||
*
|
|
||||||
* Revision 1.23 2003/09/13 00:59:02 steve
|
|
||||||
* Comments.
|
|
||||||
*
|
|
||||||
* Revision 1.22 2003/09/09 00:56:45 steve
|
|
||||||
* Reimpelement scheduler to divide nonblocking assign queue out.
|
|
||||||
*
|
|
||||||
* Revision 1.21 2003/06/17 21:28:59 steve
|
|
||||||
* Remove short int restrictions from vvp opcodes. (part 2)
|
|
||||||
*
|
|
||||||
* Revision 1.20 2003/04/01 05:32:56 steve
|
|
||||||
* Propagate output of sequential udp like non-blocksing assign.
|
|
||||||
*
|
|
||||||
* Revision 1.19 2003/03/18 01:32:33 steve
|
|
||||||
* Add the q edge flag.
|
|
||||||
*
|
|
||||||
* Revision 1.18 2003/02/09 23:33:26 steve
|
|
||||||
* Spelling fixes.
|
|
||||||
*
|
|
||||||
* Revision 1.17 2002/08/12 01:35:08 steve
|
|
||||||
* conditional ident string using autoconfig.
|
|
||||||
*
|
|
||||||
* Revision 1.16 2002/01/20 23:27:51 steve
|
|
||||||
* return calculated oval from UDP (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.15 2002/01/06 17:35:01 steve
|
|
||||||
* Feedback output, not propagated output. (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.14 2001/12/06 03:31:25 steve
|
|
||||||
* Support functor delays for gates and UDP devices.
|
|
||||||
* (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.13 2001/11/07 03:34:42 steve
|
|
||||||
* Use functor pointers where vvp_ipoint_t is unneeded.
|
|
||||||
*
|
|
||||||
* Revision 1.12 2001/10/31 04:27:47 steve
|
|
||||||
* Rewrite the functor type to have fewer functor modes,
|
|
||||||
* and use objects to manage the different types.
|
|
||||||
* (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.11 2001/09/19 04:10:40 steve
|
|
||||||
* Change UDP output only if table matches.
|
|
||||||
*
|
|
||||||
* Revision 1.10 2001/09/15 18:27:05 steve
|
|
||||||
* Make configure detect malloc.h
|
|
||||||
*
|
|
||||||
* Revision 1.9 2001/08/09 19:38:23 steve
|
|
||||||
* Nets (wires) do not use their own functors.
|
|
||||||
* Modifications to propagation of values.
|
|
||||||
* (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.8 2001/07/24 01:44:50 steve
|
|
||||||
* Fast UDP tables (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.7 2001/07/16 19:08:32 steve
|
|
||||||
* Schedule instead of propagating UDP output. (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.6 2001/06/18 00:51:23 steve
|
|
||||||
* Add more UDP edge types, and finish up compile
|
|
||||||
* and run-time support. (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.5 2001/05/31 04:12:43 steve
|
|
||||||
* Make the bufif0 and bufif1 gates strength aware,
|
|
||||||
* and accurately propagate strengths of outputs.
|
|
||||||
*
|
|
||||||
* Revision 1.4 2001/05/06 03:51:37 steve
|
|
||||||
* Regularize the mode-42 functor handling.
|
|
||||||
*
|
|
||||||
* Revision 1.3 2001/04/26 15:52:22 steve
|
|
||||||
* Add the mode-42 functor concept to UDPs.
|
|
||||||
*
|
|
||||||
* Revision 1.2 2001/04/26 03:10:55 steve
|
|
||||||
* Redo and simplify UDP behavior.
|
|
||||||
*
|
|
||||||
* Revision 1.1 2001/04/24 02:23:59 steve
|
|
||||||
* Support for UDP devices in VVP (Stephen Boettcher)
|
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
|
||||||
161
vvp/udp.h
161
vvp/udp.h
|
|
@ -1,8 +1,10 @@
|
||||||
#ifndef __udp_H
|
#ifndef __udp_H
|
||||||
#define __udp_H
|
#define __udp_H
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2001 Stephen Williams (steve@icarus.com)
|
* Copyright (c) 2005 Stephen Williams (steve@icarus.com)
|
||||||
* Copyright (c) 2001 Stephan Boettcher <stephan@nevis.columbia.edu>
|
*
|
||||||
|
* (This is a rewrite of code that was ...
|
||||||
|
* Copyright (c) 2001 Stephan Boettcher <stephan@nevis.columbia.edu>)
|
||||||
*
|
*
|
||||||
* This source code is free software; you can redistribute it
|
* This source code is free software; you can redistribute it
|
||||||
* and/or modify it in source code form under the terms of the GNU
|
* and/or modify it in source code form under the terms of the GNU
|
||||||
|
|
@ -20,106 +22,97 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: udp.h,v 1.14 2004/10/04 01:10:59 steve Exp $"
|
#ident "$Id: udp.h,v 1.15 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "functor.h"
|
# include <vvp_net.h>
|
||||||
|
|
||||||
// UDP implementation:
|
/*
|
||||||
|
* The vvp_udp_s instance represents a *definition* of a
|
||||||
|
* primitive. netlist instances refer to these definitions.
|
||||||
|
*
|
||||||
|
* The ports argument of the constructor is the number of input ports
|
||||||
|
* to the device. The single output port is not counted. The port
|
||||||
|
* count must be greater then 0.
|
||||||
|
*
|
||||||
|
* A level sensitive UDP has a table that includes all the rows that
|
||||||
|
* generate a 0 output and another table that includes all the rows
|
||||||
|
* that generate a 1 output. A set of inputs is tested against the
|
||||||
|
* entries in both sets, and if there are no matches, the output is
|
||||||
|
* set to x.
|
||||||
|
*
|
||||||
|
* The levels0 and levels1 tables are each an array of levels
|
||||||
|
* tables. Each levels table is a mask of positions that are supposed
|
||||||
|
* to be 0, 1 and x. The LSB of each mask represents first port, and
|
||||||
|
* so on. If the bit is set in mask0, a bit4_0 is expected at that
|
||||||
|
* position, and similarly for mask1 and maskx. Only exactly 1 bit
|
||||||
|
* will be set in the three masks for each bit position.
|
||||||
|
*
|
||||||
|
* This table structure implies that the number of inputs to the level
|
||||||
|
* sensitive device is limited to the number of bits in an unsigned long.
|
||||||
|
*/
|
||||||
|
|
||||||
typedef struct udp_table_entry_s *udp_table_entry_t;
|
struct udp_levels_table {
|
||||||
|
unsigned long mask0;
|
||||||
|
unsigned long mask1;
|
||||||
|
unsigned long maskx;
|
||||||
|
};
|
||||||
|
|
||||||
struct vvp_udp_s
|
class vvp_udp_s {
|
||||||
{
|
|
||||||
char *name;
|
|
||||||
udp_table_entry_t table;
|
|
||||||
unsigned ntable;
|
|
||||||
unsigned sequ;
|
|
||||||
unsigned nin;
|
|
||||||
unsigned char init;
|
|
||||||
|
|
||||||
void compile_table(char **tab);
|
public:
|
||||||
unsigned char propagate(functor_t fu, vvp_ipoint_t i);
|
vvp_udp_s(char*label, char*name, unsigned ports, bool sequ);
|
||||||
|
~vvp_udp_s();
|
||||||
|
void compile_table(char**tab);
|
||||||
|
|
||||||
|
// Return the number of input ports for the defined UDP.
|
||||||
|
unsigned port_count() const;
|
||||||
|
|
||||||
|
// Test the cur table with the compiled rows, and return the
|
||||||
|
// bit value that matches.
|
||||||
|
vvp_bit4_t test_levels(const udp_levels_table&cur);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void compile_row_(udp_table_entry_t, char *);
|
char*name_;
|
||||||
|
unsigned ports_;
|
||||||
|
|
||||||
|
// Level sensitive rows of the device.
|
||||||
|
struct udp_levels_table*levels0_;
|
||||||
|
struct udp_levels_table*levels1_;
|
||||||
|
unsigned nlevels0_, nlevels1_;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct vvp_udp_s *udp_create(char *label);
|
|
||||||
struct vvp_udp_s *udp_find(char *label);
|
|
||||||
|
|
||||||
|
|
||||||
// UDP instances:
|
|
||||||
/*
|
/*
|
||||||
* A complete UDP instance includes one udp_functor_s functor that
|
* Ths looks up a UDP definition from its LABEL.
|
||||||
* holds the output of the UDP, as well as the first 4 inputs. This
|
|
||||||
* also points to the vvp_udp_s table that is the behavior for the
|
|
||||||
* device.
|
|
||||||
*
|
|
||||||
* If there are more then 4 inputs to the device, then enough
|
|
||||||
* edge_inputs_functor_s functors is created to receive all the
|
|
||||||
* inputs. All the edge_inputs_functors_s ::out members point to the
|
|
||||||
* leading udp_functor_s object, so the ::set methods all invoke the
|
|
||||||
* ::set method of this functor.
|
|
||||||
*
|
|
||||||
* +---+ First object is a udp_functor_s object
|
|
||||||
* <----+ +--
|
|
||||||
* | +--
|
|
||||||
* | +--
|
|
||||||
* | +--
|
|
||||||
* +---+
|
|
||||||
* ^
|
|
||||||
* | +---+ Subsequent objects are edge_inputs_functor_s
|
|
||||||
* \-+ +-- that point their outputs to the leading
|
|
||||||
* ^ | +-- udp_functor_s object. There are as many
|
|
||||||
* | | +-- edge_inputs_functor_s objects as needed
|
|
||||||
* | | +-- to accommodate all the inputs of the user
|
|
||||||
* | +---+ defined primitive.
|
|
||||||
* |
|
|
||||||
* | +---+
|
|
||||||
* \-+ +--
|
|
||||||
* | +--
|
|
||||||
* | +--
|
|
||||||
* | +--
|
|
||||||
* +---+
|
|
||||||
* .
|
|
||||||
* .
|
|
||||||
* .
|
|
||||||
* (The propagate method of the vvp_udp_s object relies on the fact
|
|
||||||
* that the initial udp_functor_s and the subsequend
|
|
||||||
* edge_inputs_functor_s objects are consecutive in functor space.)
|
|
||||||
*/
|
*/
|
||||||
class udp_functor_s : public edge_inputs_functor_s
|
struct vvp_udp_s *udp_find(const char *label);
|
||||||
{
|
|
||||||
|
/*
|
||||||
|
* The udp_fun_core is the core of the udp instance in the
|
||||||
|
* netlist. This core is a vvp_wide_fun_core that takes care of
|
||||||
|
* dispatching the output. This class also receives the inputs from
|
||||||
|
* the vvp_wide_fun_t objects and processes them to generate the
|
||||||
|
* output to be sent.
|
||||||
|
*/
|
||||||
|
class vvp_udp_fun_core : public vvp_wide_fun_core {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit udp_functor_s(vvp_udp_s *u) : udp(u) {}
|
vvp_udp_fun_core(vvp_net_t*net, vvp_udp_s*def);
|
||||||
void set(vvp_ipoint_t i, bool push, unsigned val, unsigned str);
|
~vvp_udp_fun_core();
|
||||||
vvp_udp_s *udp;
|
|
||||||
|
void recv_vec4_from_inputs(unsigned);
|
||||||
|
|
||||||
|
private:
|
||||||
|
vvp_udp_s*def_;
|
||||||
|
udp_levels_table current_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: udp.h,v $
|
* $Log: udp.h,v $
|
||||||
* Revision 1.14 2004/10/04 01:10:59 steve
|
* Revision 1.15 2005/04/01 06:02:45 steve
|
||||||
* Clean up spurious trailing white space.
|
* Reimplement combinational UDPs.
|
||||||
*
|
*
|
||||||
* Revision 1.13 2003/09/17 03:39:55 steve
|
|
||||||
* Internal documentation of UDP devices.
|
|
||||||
*
|
|
||||||
* Revision 1.12 2003/06/17 21:28:59 steve
|
|
||||||
* Remove short int restrictions from vvp opcodes. (part 2)
|
|
||||||
*
|
|
||||||
* Revision 1.11 2002/08/12 01:35:08 steve
|
|
||||||
* conditional ident string using autoconfig.
|
|
||||||
*
|
|
||||||
* Revision 1.10 2001/10/31 04:27:47 steve
|
|
||||||
* Rewrite the functor type to have fewer functor modes,
|
|
||||||
* and use objects to manage the different types.
|
|
||||||
* (Stephan Boettcher)
|
|
||||||
*
|
|
||||||
* Revision 1.9 2001/08/09 19:38:23 steve
|
|
||||||
* Nets (wires) do not use their own functors.
|
|
||||||
* Modifications to propagation of values.
|
|
||||||
* (Stephan Boettcher)
|
|
||||||
*/
|
*/
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
54
vvp/ufunc.cc
54
vvp/ufunc.cc
|
|
@ -17,7 +17,7 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: ufunc.cc,v 1.6 2005/03/18 02:56:04 steve Exp $"
|
#ident "$Id: ufunc.cc,v 1.7 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
# include "compile.h"
|
# include "compile.h"
|
||||||
|
|
@ -43,12 +43,10 @@ ufunc_core::ufunc_core(unsigned owid, vvp_net_t*ptr,
|
||||||
unsigned nports, vvp_net_t**ports,
|
unsigned nports, vvp_net_t**ports,
|
||||||
vvp_code_t sa, struct __vpiScope*run_scope,
|
vvp_code_t sa, struct __vpiScope*run_scope,
|
||||||
char*result_label)
|
char*result_label)
|
||||||
|
: vvp_wide_fun_core(ptr, nports)
|
||||||
{
|
{
|
||||||
owid_ = owid;
|
owid_ = owid;
|
||||||
onet_ = ptr;
|
|
||||||
nports_ = nports;
|
|
||||||
ports_ = ports;
|
ports_ = ports;
|
||||||
port_values_ = new vvp_vector4_t[nports];
|
|
||||||
code_ = sa;
|
code_ = sa;
|
||||||
thread_ = 0;
|
thread_ = 0;
|
||||||
scope_ = run_scope;
|
scope_ = run_scope;
|
||||||
|
|
@ -58,7 +56,6 @@ ufunc_core::ufunc_core(unsigned owid, vvp_net_t*ptr,
|
||||||
|
|
||||||
ufunc_core::~ufunc_core()
|
ufunc_core::~ufunc_core()
|
||||||
{
|
{
|
||||||
delete[]port_values_;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
@ -68,10 +65,10 @@ ufunc_core::~ufunc_core()
|
||||||
*/
|
*/
|
||||||
void ufunc_core::assign_bits_to_ports(void)
|
void ufunc_core::assign_bits_to_ports(void)
|
||||||
{
|
{
|
||||||
for (unsigned idx = 0 ; idx < nports_ ; idx += 1) {
|
for (unsigned idx = 0 ; idx < port_count() ; idx += 1) {
|
||||||
vvp_net_t*net = ports_[idx];
|
vvp_net_t*net = ports_[idx];
|
||||||
vvp_net_ptr_t pp (net, 0);
|
vvp_net_ptr_t pp (net, 0);
|
||||||
net->fun->recv_vec4(pp, port_values_[idx]);
|
net->fun->recv_vec4(pp, value(idx));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -83,10 +80,8 @@ void ufunc_core::assign_bits_to_ports(void)
|
||||||
void ufunc_core::finish_thread(vthread_t thr)
|
void ufunc_core::finish_thread(vthread_t thr)
|
||||||
{
|
{
|
||||||
thread_ = 0;
|
thread_ = 0;
|
||||||
|
|
||||||
vvp_fun_signal*sig = reinterpret_cast<vvp_fun_signal*>(result_->fun);
|
vvp_fun_signal*sig = reinterpret_cast<vvp_fun_signal*>(result_->fun);
|
||||||
|
propagate_vec4(sig->vec4_value());
|
||||||
vvp_send_vec4(onet_->out, sig->vec4_value());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
@ -94,32 +89,14 @@ void ufunc_core::finish_thread(vthread_t thr)
|
||||||
* input value to the port of the functor. I save the input value and
|
* input value to the port of the functor. I save the input value and
|
||||||
* arrange for the function to be called.
|
* arrange for the function to be called.
|
||||||
*/
|
*/
|
||||||
void ufunc_core::recv_vec4_from_inputs(unsigned port, vvp_vector4_t bit)
|
void ufunc_core::recv_vec4_from_inputs(unsigned port)
|
||||||
{
|
{
|
||||||
assert(port < nports_);
|
|
||||||
port_values_[port] = bit;
|
|
||||||
|
|
||||||
if (thread_ == 0) {
|
if (thread_ == 0) {
|
||||||
thread_ = vthread_new(code_, scope_);
|
thread_ = vthread_new(code_, scope_);
|
||||||
schedule_vthread(thread_, 0);
|
schedule_vthread(thread_, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ufunc_input_functor::ufunc_input_functor(ufunc_core*c, unsigned base)
|
|
||||||
: core_(c), port_base_(base)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
ufunc_input_functor::~ufunc_input_functor()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void ufunc_input_functor::recv_vec4(vvp_net_ptr_t port, vvp_vector4_t bit)
|
|
||||||
{
|
|
||||||
unsigned pidx = port_base_ + port.port();
|
|
||||||
core_->recv_vec4_from_inputs(pidx, bit);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function compiles the .ufunc statement that is discovered in
|
* This function compiles the .ufunc statement that is discovered in
|
||||||
* the source file. Create all the functors and the thread, and
|
* the source file. Create all the functors and the thread, and
|
||||||
|
|
@ -198,21 +175,7 @@ void compile_ufunc(char*label, char*code, unsigned wid,
|
||||||
start_code->ufunc_core_ptr = fcore;
|
start_code->ufunc_core_ptr = fcore;
|
||||||
ujoin_code->ufunc_core_ptr = fcore;
|
ujoin_code->ufunc_core_ptr = fcore;
|
||||||
|
|
||||||
/* Create input functors to receive values from the
|
wide_inputs_connect(fcore, argc, argv);
|
||||||
network. These functors pass the data to the core. */
|
|
||||||
unsigned input_functors = (argc+3) / 4;
|
|
||||||
for (unsigned idx = 0 ; idx < input_functors ; idx += 1) {
|
|
||||||
unsigned base = idx*4;
|
|
||||||
unsigned trans = 4;
|
|
||||||
if (base+trans > argc)
|
|
||||||
trans = argc - base;
|
|
||||||
|
|
||||||
ufunc_input_functor*cur = new ufunc_input_functor(fcore, base);
|
|
||||||
ptr = new vvp_net_t;
|
|
||||||
ptr->fun = cur;
|
|
||||||
|
|
||||||
inputs_connect(ptr, trans, argv+base);
|
|
||||||
}
|
|
||||||
|
|
||||||
free(argv);
|
free(argv);
|
||||||
free(portv);
|
free(portv);
|
||||||
|
|
@ -221,6 +184,9 @@ void compile_ufunc(char*label, char*code, unsigned wid,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: ufunc.cc,v $
|
* $Log: ufunc.cc,v $
|
||||||
|
* Revision 1.7 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.6 2005/03/18 02:56:04 steve
|
* Revision 1.6 2005/03/18 02:56:04 steve
|
||||||
* Add support for LPM_UFUNC user defined functions.
|
* Add support for LPM_UFUNC user defined functions.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
22
vvp/ufunc.h
22
vvp/ufunc.h
|
|
@ -19,7 +19,7 @@
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ifdef HAVE_CVS_IDENT
|
#ifdef HAVE_CVS_IDENT
|
||||||
#ident "$Id: ufunc.h,v 1.4 2005/03/18 02:56:04 steve Exp $"
|
#ident "$Id: ufunc.h,v 1.5 2005/04/01 06:02:45 steve Exp $"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
# include "pointers.h"
|
# include "pointers.h"
|
||||||
|
|
@ -46,9 +46,11 @@
|
||||||
* ufunc_core is also a functor whose output is connected to the rest
|
* ufunc_core is also a functor whose output is connected to the rest
|
||||||
* of the netlist. This is where the result is delivered back to the
|
* of the netlist. This is where the result is delivered back to the
|
||||||
* netlist.
|
* netlist.
|
||||||
|
*
|
||||||
|
* This class relies to the vvp_wide_fun_* classes in vvp_net.h.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class ufunc_core : public vvp_net_fun_t {
|
class ufunc_core : public vvp_wide_fun_core {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ufunc_core(unsigned ow, vvp_net_t*ptr,
|
ufunc_core(unsigned ow, vvp_net_t*ptr,
|
||||||
|
|
@ -64,18 +66,15 @@ class ufunc_core : public vvp_net_fun_t {
|
||||||
void finish_thread(vthread_t thr);
|
void finish_thread(vthread_t thr);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class ufunc_input_functor;
|
void recv_vec4_from_inputs(unsigned port);
|
||||||
void recv_vec4_from_inputs(unsigned port, vvp_vector4_t bit);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// output width of the function node.
|
// output width of the function node.
|
||||||
unsigned owid_;
|
unsigned owid_;
|
||||||
// Pointer to the "self" vvp_net_t that points to this functor.
|
// The vvp_net_t* objects for the function input ports. We use
|
||||||
vvp_net_t*onet_;
|
// these to write the input values to the reg input variable
|
||||||
// Structure to track the input values from the input functors.
|
// functors for the thread.
|
||||||
unsigned nports_;
|
|
||||||
vvp_net_t**ports_;
|
vvp_net_t**ports_;
|
||||||
vvp_vector4_t*port_values_;
|
|
||||||
|
|
||||||
// This is a thread to execute the behavioral portion of the
|
// This is a thread to execute the behavioral portion of the
|
||||||
// function.
|
// function.
|
||||||
|
|
@ -87,6 +86,7 @@ class ufunc_core : public vvp_net_fun_t {
|
||||||
vvp_net_t*result_;
|
vvp_net_t*result_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if 0
|
||||||
/*
|
/*
|
||||||
* The job of the input functor is only to monitor inputs to the
|
* The job of the input functor is only to monitor inputs to the
|
||||||
* function and pass them to the ufunc_core object. Each functor takes
|
* function and pass them to the ufunc_core object. Each functor takes
|
||||||
|
|
@ -105,9 +105,13 @@ class ufunc_input_functor : public vvp_net_fun_t {
|
||||||
ufunc_core*core_;
|
ufunc_core*core_;
|
||||||
unsigned port_base_;
|
unsigned port_base_;
|
||||||
};
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: ufunc.h,v $
|
* $Log: ufunc.h,v $
|
||||||
|
* Revision 1.5 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.4 2005/03/18 02:56:04 steve
|
* Revision 1.4 2005/03/18 02:56:04 steve
|
||||||
* Add support for LPM_UFUNC user defined functions.
|
* Add support for LPM_UFUNC user defined functions.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
|
|
@ -16,7 +16,7 @@
|
||||||
* along with this program; if not, write to the Free Software
|
* along with this program; if not, write to the Free Software
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ident "$Id: vvp_net.cc,v 1.19 2005/03/18 02:56:04 steve Exp $"
|
#ident "$Id: vvp_net.cc,v 1.20 2005/04/01 06:02:45 steve Exp $"
|
||||||
|
|
||||||
# include "vvp_net.h"
|
# include "vvp_net.h"
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
|
|
@ -781,6 +781,70 @@ vvp_vector4_t vvp_fun_signal::vec4_value() const
|
||||||
return bits4_;
|
return bits4_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* **** vvp_wide_fun_* methods **** */
|
||||||
|
|
||||||
|
vvp_wide_fun_core::vvp_wide_fun_core(vvp_net_t*net, unsigned nports)
|
||||||
|
{
|
||||||
|
delay_ = 0;
|
||||||
|
ptr_ = net;
|
||||||
|
nports_ = nports;
|
||||||
|
port_values_ = new vvp_vector4_t [nports_];
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_wide_fun_core::~vvp_wide_fun_core()
|
||||||
|
{
|
||||||
|
delete[]port_values_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_wide_fun_core::set_output_delay(vvp_time64_t delay)
|
||||||
|
{
|
||||||
|
delay_ = delay;
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_wide_fun_core::propagate_vec4(const vvp_vector4_t&bit)
|
||||||
|
{
|
||||||
|
if (delay_)
|
||||||
|
schedule_assign_vector(ptr_->out, bit, delay_);
|
||||||
|
else
|
||||||
|
vvp_send_vec4(ptr_->out, bit);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned vvp_wide_fun_core::port_count() const
|
||||||
|
{
|
||||||
|
return nports_;
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_vector4_t& vvp_wide_fun_core::value(unsigned idx)
|
||||||
|
{
|
||||||
|
assert(idx < nports_);
|
||||||
|
return port_values_[idx];
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_wide_fun_core::dispatch_vec4_from_input_(unsigned port,
|
||||||
|
vvp_vector4_t bit)
|
||||||
|
{
|
||||||
|
assert(port < nports_);
|
||||||
|
port_values_[port] = bit;
|
||||||
|
recv_vec4_from_inputs(port);
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_wide_fun_t::vvp_wide_fun_t(vvp_wide_fun_core*c, unsigned base)
|
||||||
|
: core_(c), port_base_(base)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
vvp_wide_fun_t::~vvp_wide_fun_t()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void vvp_wide_fun_t::recv_vec4(vvp_net_ptr_t port, vvp_vector4_t bit)
|
||||||
|
{
|
||||||
|
unsigned pidx = port_base_ + port.port();
|
||||||
|
core_->dispatch_vec4_from_input_(pidx, bit);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* **** vvp_scalar_t methods **** */
|
/* **** vvp_scalar_t methods **** */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
@ -1178,6 +1242,9 @@ vvp_bit4_t compare_gtge_signed(const vvp_vector4_t&a,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: vvp_net.cc,v $
|
* $Log: vvp_net.cc,v $
|
||||||
|
* Revision 1.20 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.19 2005/03/18 02:56:04 steve
|
* Revision 1.19 2005/03/18 02:56:04 steve
|
||||||
* Add support for LPM_UFUNC user defined functions.
|
* Add support for LPM_UFUNC user defined functions.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
|
|
@ -18,7 +18,7 @@
|
||||||
* along with this program; if not, write to the Free Software
|
* along with this program; if not, write to the Free Software
|
||||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
||||||
*/
|
*/
|
||||||
#ident "$Id: vvp_net.h,v 1.19 2005/03/18 02:56:04 steve Exp $"
|
#ident "$Id: vvp_net.h,v 1.20 2005/04/01 06:02:45 steve Exp $"
|
||||||
|
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
# include <assert.h>
|
# include <assert.h>
|
||||||
|
|
@ -602,8 +602,86 @@ class vvp_fun_signal : public vvp_net_fun_t {
|
||||||
void run_vpi_callbacks();
|
void run_vpi_callbacks();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Wide Functors:
|
||||||
|
* Wide functors represent special devices that may have more then 4
|
||||||
|
* input ports. These devices need a set of N/4 actual functors to
|
||||||
|
* catch the inputs, and use another to deliver the output.
|
||||||
|
*
|
||||||
|
* vvp_wide_fun_t --+--> vvp_wide_fun_core --> ...
|
||||||
|
* |
|
||||||
|
* vvp_wide_fun_t --+
|
||||||
|
* |
|
||||||
|
* vvp_wide_fun_t --+
|
||||||
|
*
|
||||||
|
* There are enough input functors to take all the functor inputs, 4
|
||||||
|
* per functor. These inputs deliver the changed input value to the
|
||||||
|
* wide_fun_core, which carries the infastructure for the thread. The
|
||||||
|
* wide_fun_core is also a functor whose output is connected to the rest
|
||||||
|
* of the netlist. This is where the result is delivered back to the
|
||||||
|
* netlist.
|
||||||
|
*
|
||||||
|
* The vvp_wide_fun_core keeps a store of the inputs from all the
|
||||||
|
* input functors, and makes them available to the derived class that
|
||||||
|
* does the processing.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class vvp_wide_fun_core : public vvp_net_fun_t {
|
||||||
|
|
||||||
|
public:
|
||||||
|
vvp_wide_fun_core(vvp_net_t*net, unsigned nports);
|
||||||
|
virtual ~vvp_wide_fun_core();
|
||||||
|
|
||||||
|
void set_output_delay(vvp_time64_t delay);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void propagate_vec4(const vvp_vector4_t&bit);
|
||||||
|
unsigned port_count() const;
|
||||||
|
vvp_vector4_t& value(unsigned);
|
||||||
|
|
||||||
|
private:
|
||||||
|
// the derived class implements this to receive an indication
|
||||||
|
// that one of the port input values changed.
|
||||||
|
virtual void recv_vec4_from_inputs(unsigned port) =0;
|
||||||
|
|
||||||
|
friend class vvp_wide_fun_t;
|
||||||
|
void dispatch_vec4_from_input_(unsigned port, vvp_vector4_t bit);
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Propagation delay, if any, for the output from the device.
|
||||||
|
vvp_time64_t delay_;
|
||||||
|
// Back-point to the vvp_net_t that points to me.
|
||||||
|
vvp_net_t*ptr_;
|
||||||
|
// Structure to track the input values from the input functors.
|
||||||
|
unsigned nports_;
|
||||||
|
vvp_vector4_t*port_values_;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The job of the input functor is only to monitor inputs to the
|
||||||
|
* function and pass them to the ufunc_core object. Each functor takes
|
||||||
|
* up to 4 inputs, with the base the port number for the first
|
||||||
|
* function input that this functor represents.
|
||||||
|
*/
|
||||||
|
class vvp_wide_fun_t : public vvp_net_fun_t {
|
||||||
|
|
||||||
|
public:
|
||||||
|
vvp_wide_fun_t(vvp_wide_fun_core*c, unsigned base);
|
||||||
|
~vvp_wide_fun_t();
|
||||||
|
|
||||||
|
void recv_vec4(vvp_net_ptr_t port, vvp_vector4_t bit);
|
||||||
|
|
||||||
|
private:
|
||||||
|
vvp_wide_fun_core*core_;
|
||||||
|
unsigned port_base_;
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $Log: vvp_net.h,v $
|
* $Log: vvp_net.h,v $
|
||||||
|
* Revision 1.20 2005/04/01 06:02:45 steve
|
||||||
|
* Reimplement combinational UDPs.
|
||||||
|
*
|
||||||
* Revision 1.19 2005/03/18 02:56:04 steve
|
* Revision 1.19 2005/03/18 02:56:04 steve
|
||||||
* Add support for LPM_UFUNC user defined functions.
|
* Add support for LPM_UFUNC user defined functions.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue