2001-05-01 03:09:39 +02:00
|
|
|
/*
|
2005-03-03 05:33:10 +01:00
|
|
|
* Copyright (c) 2005 Stephen Williams (steve@icarus.com)
|
2001-05-01 03:09:39 +02:00
|
|
|
* Copyright (c) 2000 Stephen Williams (steve@icarus.com)
|
|
|
|
|
* Copyright (c) 2001 Stephan Boettcher <stephan@nevis.columbia.edu>
|
|
|
|
|
*
|
|
|
|
|
* This source code is free software; you can redistribute it
|
|
|
|
|
* and/or modify it in source code form under the terms of the GNU
|
|
|
|
|
* General Public License as published by the Free Software
|
|
|
|
|
* Foundation; either version 2 of the License, or (at your option)
|
|
|
|
|
* any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
|
|
|
*/
|
2002-08-12 03:34:58 +02:00
|
|
|
#ifdef HAVE_CVS_IDENT
|
2005-03-03 05:33:10 +01:00
|
|
|
#ident "$Id: memory.cc,v 1.23 2005/03/03 04:33:10 steve Exp $"
|
2001-05-01 03:09:39 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include "memory.h"
|
|
|
|
|
#include "symbols.h"
|
|
|
|
|
#include "schedule.h"
|
|
|
|
|
#include <assert.h>
|
2001-09-15 20:27:04 +02:00
|
|
|
#ifdef HAVE_MALLOC_H
|
2001-05-01 03:09:39 +02:00
|
|
|
#include <malloc.h>
|
2001-09-15 20:27:04 +02:00
|
|
|
#endif
|
|
|
|
|
#include <stdlib.h>
|
2001-05-01 03:09:39 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct vvp_memory_port_s *vvp_memory_port_t;
|
|
|
|
|
|
|
|
|
|
struct vvp_memory_s
|
|
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
// Address ranges (1 or more)
|
|
|
|
|
unsigned nrange;
|
|
|
|
|
struct memory_address_range*ranges;
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
// Data port properties:
|
|
|
|
|
unsigned width; // number of data bits
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
int msb, lsb; // Most/Least Significant data bit (VPI)
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
// Array of words.
|
|
|
|
|
unsigned word_count;
|
|
|
|
|
vvp_vector4_t*words;
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
// List of ports into this memory.
|
|
|
|
|
vvp_memory_port_t port_list;
|
|
|
|
|
};
|
2001-05-01 03:09:39 +02:00
|
|
|
|
|
|
|
|
#define VVP_MEMORY_NO_ADDR ((int)0x80000000)
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-10-31 05:27:46 +01:00
|
|
|
struct vvp_memory_port_s : public functor_s
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2001-10-31 05:27:46 +01:00
|
|
|
void set(vvp_ipoint_t i, bool push, unsigned val, unsigned str);
|
2001-05-06 05:51:37 +02:00
|
|
|
|
|
|
|
|
vvp_memory_t mem;
|
|
|
|
|
vvp_ipoint_t ix;
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
unsigned naddr;
|
|
|
|
|
|
2001-05-06 05:51:37 +02:00
|
|
|
vvp_memory_port_t next;
|
|
|
|
|
int cur_addr;
|
|
|
|
|
vvp_memory_bits_t cur_bits;
|
2001-06-15 05:28:30 +02:00
|
|
|
unsigned bitoff;
|
|
|
|
|
unsigned nbits;
|
|
|
|
|
|
|
|
|
|
bool writable;
|
2001-05-01 03:09:39 +02:00
|
|
|
};
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
// Compilation
|
|
|
|
|
|
|
|
|
|
static symbol_table_t memory_table = 0;
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
vvp_memory_t memory_find(char *label)
|
2001-05-09 01:59:33 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
if (memory_table == 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
symbol_value_t v = sym_get_value(memory_table, label);
|
|
|
|
|
return (vvp_memory_t)v.ptr;
|
2001-05-09 01:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
vvp_memory_t memory_create(char *label)
|
2001-05-09 01:59:33 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
if (!memory_table)
|
|
|
|
|
memory_table = new_symbol_table();
|
|
|
|
|
|
|
|
|
|
assert(!memory_find(label));
|
|
|
|
|
|
|
|
|
|
vvp_memory_t mem = new struct vvp_memory_s;
|
|
|
|
|
|
|
|
|
|
symbol_value_t v;
|
|
|
|
|
v.ptr = mem;
|
|
|
|
|
sym_set_value(memory_table, label, v);
|
|
|
|
|
|
|
|
|
|
return mem;
|
2001-05-09 01:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
void memory_configure(vvp_memory_t mem,
|
|
|
|
|
int msb, int lsb,
|
|
|
|
|
unsigned nrange,
|
|
|
|
|
const struct memory_address_range*ranges)
|
2002-01-31 05:28:17 +01:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
/* Get the word width details. */
|
|
|
|
|
mem->width = msb > lsb ? msb-lsb+1 : lsb-msb+1;
|
|
|
|
|
mem->msb = msb;
|
|
|
|
|
mem->lsb = lsb;
|
|
|
|
|
|
|
|
|
|
/* Make a private copy of the memory address ranges. */
|
|
|
|
|
assert(nrange > 0);
|
|
|
|
|
mem->nrange = nrange;
|
|
|
|
|
mem->ranges = new struct memory_address_range[nrange];
|
|
|
|
|
for (unsigned idx = 0 ; idx < nrange ; idx += 1)
|
|
|
|
|
mem->ranges[idx] = ranges[idx];
|
|
|
|
|
|
|
|
|
|
/* Scan the indices (multiplying each range) to add up the
|
|
|
|
|
total number of words in this memory. */
|
|
|
|
|
mem->word_count = 1;
|
|
|
|
|
for (unsigned idx = 0 ; idx < mem->nrange ; idx += 1) {
|
|
|
|
|
struct memory_address_range*rp = mem->ranges+idx;
|
|
|
|
|
|
|
|
|
|
unsigned count = rp->msb > rp->lsb
|
|
|
|
|
? rp->msb - rp->lsb + 1
|
|
|
|
|
: rp->lsb - rp->msb + 1;
|
|
|
|
|
|
|
|
|
|
mem->word_count *= count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mem->words = new vvp_vector4_t [mem->word_count];
|
|
|
|
|
assert(mem->words);
|
|
|
|
|
|
|
|
|
|
mem->port_list = 0;
|
2002-01-31 05:28:17 +01:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
unsigned memory_word_width(vvp_memory_t mem)
|
2002-01-31 05:28:17 +01:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
return mem->width;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned memory_word_count(vvp_memory_t mem)
|
|
|
|
|
{
|
|
|
|
|
return mem->word_count;
|
2002-01-31 05:28:17 +01:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
long memory_word_left_range(vvp_memory_t mem)
|
2002-01-31 05:28:17 +01:00
|
|
|
{
|
|
|
|
|
return mem->msb;
|
|
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
long memory_word_right_range(vvp_memory_t mem)
|
2002-01-31 05:28:17 +01:00
|
|
|
{
|
|
|
|
|
return mem->lsb;
|
|
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
long memory_left_range(vvp_memory_t mem, unsigned ix)
|
2001-05-09 01:59:33 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
assert(ix < mem->nrange);
|
|
|
|
|
return mem->ranges[ix].msb;
|
2001-05-09 01:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
long memory_right_range(vvp_memory_t mem, unsigned ix)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
assert(ix < mem->nrange);
|
|
|
|
|
return mem->ranges[ix].lsb;
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
vvp_vector4_t memory_get_word(vvp_memory_t mem, unsigned addr)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
// XXXX For now, assume this can't happen
|
|
|
|
|
assert(addr <= mem->word_count);
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
return mem->words[addr];
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
void memory_init_word(vvp_memory_t mem, unsigned addr, vvp_vector4_t val)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
if (addr >= mem->word_count)
|
|
|
|
|
return;
|
2002-01-31 05:28:17 +01:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
mem->words[addr] = val;
|
|
|
|
|
}
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
void memory_set_word(vvp_memory_t mem, unsigned addr, vvp_vector4_t val)
|
|
|
|
|
{
|
|
|
|
|
memory_init_word(mem, addr, val);
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
if (mem->port_list)
|
|
|
|
|
fprintf(stderr, "XXXX memory_set_word(%u, ...)"
|
|
|
|
|
" not fully implemented\n", addr);
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-10-31 05:27:46 +01:00
|
|
|
vvp_ipoint_t memory_port_new(vvp_memory_t mem,
|
|
|
|
|
unsigned nbits, unsigned bitoff,
|
|
|
|
|
unsigned naddr, bool writable)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2001-10-31 05:27:46 +01:00
|
|
|
unsigned nfun = naddr;
|
|
|
|
|
if (writable)
|
|
|
|
|
nfun += 2 + nbits;
|
|
|
|
|
nfun = (nfun+3)/4;
|
|
|
|
|
if (nfun < nbits)
|
|
|
|
|
nfun = nbits;
|
|
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
vvp_memory_port_t a = new struct vvp_memory_port_s;
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
a->mem = mem;
|
2001-06-15 05:28:30 +02:00
|
|
|
a->naddr = naddr;
|
|
|
|
|
a->writable = writable;
|
2001-05-01 03:09:39 +02:00
|
|
|
a->nbits = nbits;
|
|
|
|
|
a->bitoff = bitoff;
|
|
|
|
|
a->next = mem->addr_root;
|
2001-08-09 21:37:05 +02:00
|
|
|
mem->addr_root = a;
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-10-31 05:27:46 +01:00
|
|
|
a->ix = functor_allocate(nfun);
|
|
|
|
|
functor_define(a->ix, a);
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2004-10-04 03:10:51 +02:00
|
|
|
if (nfun > 1)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2001-10-31 05:27:46 +01:00
|
|
|
extra_ports_functor_s *fu = new extra_ports_functor_s[nfun-1];
|
|
|
|
|
for (unsigned i = 0; i< nfun - 1; i++) {
|
|
|
|
|
fu[i].base_ = a->ix;
|
|
|
|
|
functor_define(ipoint_index(a->ix, i+1), fu+i);
|
|
|
|
|
}
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
a->cur_addr = VVP_MEMORY_NO_ADDR;
|
|
|
|
|
a->cur_bits = 0x0;
|
2001-09-11 03:54:58 +02:00
|
|
|
|
2001-10-31 05:27:46 +01:00
|
|
|
return a->ix;
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
void schedule_memory(vvp_memory_t mem, unsigned addr,
|
|
|
|
|
vvp_vector4_t val, unsigned long delay)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2005-03-03 05:33:10 +01:00
|
|
|
fprintf(stderr, "XXXX Forgot how to schedule memory write.\n");
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Utilities
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2004-10-04 03:10:51 +02:00
|
|
|
inline static
|
2001-10-14 03:36:12 +02:00
|
|
|
vvp_memory_bits_t get_word_ix(vvp_memory_t mem, unsigned idx)
|
|
|
|
|
{
|
|
|
|
|
return mem->bits + idx*mem->fwidth;
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2004-10-04 03:10:51 +02:00
|
|
|
inline static
|
2001-05-01 03:09:39 +02:00
|
|
|
vvp_memory_bits_t get_word(vvp_memory_t mem, int addr)
|
|
|
|
|
{
|
2001-06-15 05:28:30 +02:00
|
|
|
assert(mem->a_idxs==1);
|
|
|
|
|
unsigned waddr = addr - mem->a_idx[0].first;
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
if (waddr >= mem->size)
|
|
|
|
|
return 0x0;
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-10-14 03:36:12 +02:00
|
|
|
return get_word_ix(mem, waddr);
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
inline static
|
|
|
|
|
bool set_bit(vvp_memory_bits_t bits, int bit, unsigned char val)
|
|
|
|
|
{
|
|
|
|
|
int ix = bit/4;
|
|
|
|
|
int ip = 2*(bit%4);
|
|
|
|
|
bool r = ((bits[ix] >> ip) & 3) != val;
|
|
|
|
|
bits[ix] = (bits[ix] &~ (3<<ip)) | ((val&3) << ip);
|
|
|
|
|
return r;
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
inline static
|
|
|
|
|
unsigned char get_nibble(vvp_memory_bits_t bits, int bit)
|
|
|
|
|
{
|
|
|
|
|
if (!bits)
|
|
|
|
|
return 0xaa;
|
|
|
|
|
int ix = bit/4;
|
|
|
|
|
return bits[ix];
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
inline static
|
|
|
|
|
unsigned char get_bit(vvp_memory_bits_t bits, int bit)
|
|
|
|
|
{
|
|
|
|
|
return (get_nibble(bits, bit) >> (2*(bit&3))) & 3;
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2004-10-04 03:10:51 +02:00
|
|
|
inline static
|
2001-05-01 03:09:39 +02:00
|
|
|
unsigned char functor_get_inputs(vvp_ipoint_t ip)
|
|
|
|
|
{
|
|
|
|
|
functor_t fp = functor_index(ip);
|
|
|
|
|
assert(fp);
|
|
|
|
|
return fp->ival;
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2004-10-04 03:10:51 +02:00
|
|
|
inline static
|
2001-05-01 03:09:39 +02:00
|
|
|
unsigned char functor_get_input(vvp_ipoint_t ip)
|
|
|
|
|
{
|
|
|
|
|
unsigned char bits = functor_get_inputs(ip);
|
|
|
|
|
return (bits >> (2*ipoint_port(ip))) & 3;
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
|
|
|
|
#if 0
|
2004-10-04 03:10:51 +02:00
|
|
|
static
|
2001-05-01 03:09:39 +02:00
|
|
|
bool update_addr_bit(vvp_memory_port_t addr, vvp_ipoint_t ip)
|
|
|
|
|
{
|
|
|
|
|
unsigned abit = ip - addr->ix;
|
|
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
assert(abit >= 0 && abit < addr->naddr);
|
2001-05-01 03:09:39 +02:00
|
|
|
|
|
|
|
|
int old = addr->cur_addr;
|
|
|
|
|
|
|
|
|
|
int abval = functor_get_input(ip);
|
|
|
|
|
if (abval>1)
|
|
|
|
|
addr->cur_addr = VVP_MEMORY_NO_ADDR;
|
|
|
|
|
else if (addr->cur_addr == VVP_MEMORY_NO_ADDR)
|
|
|
|
|
update_addr(addr);
|
|
|
|
|
else if (abval)
|
|
|
|
|
addr->cur_addr |= (1<<abit);
|
|
|
|
|
else
|
|
|
|
|
addr->cur_addr &=~ (1<<abit);
|
|
|
|
|
|
2001-10-14 03:36:12 +02:00
|
|
|
addr->cur_bits = get_word(addr->mem, addr->cur_addr);
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2001-10-14 03:36:12 +02:00
|
|
|
return addr->cur_addr != old;
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
static
|
|
|
|
|
void update_addr(vvp_memory_port_t addr)
|
|
|
|
|
{
|
2001-08-09 21:37:05 +02:00
|
|
|
addr->cur_addr = 0;
|
2001-06-15 05:28:30 +02:00
|
|
|
for (unsigned i=0; i < addr->naddr; i++)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
|
|
|
|
update_addr_bit(addr, addr->ix+i);
|
|
|
|
|
if (addr->cur_addr == VVP_MEMORY_NO_ADDR)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
inline static
|
2001-10-14 03:36:12 +02:00
|
|
|
void update_data(vvp_memory_port_t data)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
|
|
|
|
assert(data);
|
2001-06-15 05:28:30 +02:00
|
|
|
for (unsigned i=0; i < data->nbits; i++)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
|
|
|
|
vvp_ipoint_t dx = ipoint_index(data->ix, i);
|
|
|
|
|
functor_t df = functor_index(dx);
|
2001-10-14 03:36:12 +02:00
|
|
|
unsigned char out = get_bit(data->cur_bits, i + data->bitoff);
|
2002-03-17 04:23:10 +01:00
|
|
|
df->put_oval(out, true);
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
static
|
2004-10-04 03:10:51 +02:00
|
|
|
void update_data_ports(vvp_memory_t mem, vvp_memory_bits_t bits, int bit,
|
2001-05-01 03:09:39 +02:00
|
|
|
unsigned char val)
|
|
|
|
|
{
|
2001-10-14 03:36:12 +02:00
|
|
|
if (!bits)
|
|
|
|
|
return;
|
|
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
vvp_memory_port_t a = mem->addr_root;
|
|
|
|
|
while (a)
|
|
|
|
|
{
|
2001-10-14 03:36:12 +02:00
|
|
|
if (bits == a->cur_bits)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2001-06-15 05:28:30 +02:00
|
|
|
unsigned i = bit - a->bitoff;
|
|
|
|
|
if (i < a->nbits)
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
2001-06-15 05:28:30 +02:00
|
|
|
vvp_ipoint_t ix = ipoint_index(a->ix, i);
|
2001-05-01 03:09:39 +02:00
|
|
|
functor_t df = functor_index(ix);
|
2002-03-17 04:23:10 +01:00
|
|
|
df->put_oval(val, true);
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-06-15 05:28:30 +02:00
|
|
|
static inline
|
|
|
|
|
void write_event(vvp_memory_port_t p)
|
|
|
|
|
{
|
2001-10-14 03:36:12 +02:00
|
|
|
if (!p->cur_bits)
|
|
|
|
|
return;
|
|
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
unsigned we = functor_get_input(p->ix + p->naddr + 1);
|
|
|
|
|
if (!we)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
for (unsigned i=0; i < p->nbits; i++)
|
|
|
|
|
{
|
|
|
|
|
unsigned val = functor_get_input(p->ix + p->naddr + 2 + i);
|
|
|
|
|
if (set_bit(p->cur_bits, i + p->bitoff, val))
|
|
|
|
|
{
|
2004-10-04 03:10:51 +02:00
|
|
|
// if a write would change the memory bit, but <we> is
|
2001-06-15 05:28:30 +02:00
|
|
|
// undefined (x or z), set the bit to x.
|
|
|
|
|
if (we > 1)
|
|
|
|
|
{
|
|
|
|
|
set_bit(p->cur_bits, i + p->bitoff, 2);
|
|
|
|
|
val = 2;
|
|
|
|
|
}
|
2001-10-14 03:36:12 +02:00
|
|
|
update_data_ports(p->mem, p->cur_bits, i + p->bitoff, val);
|
2001-06-15 05:28:30 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-06-15 05:28:30 +02:00
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-10-31 05:27:46 +01:00
|
|
|
void vvp_memory_port_s::set(vvp_ipoint_t i, bool, unsigned val, unsigned)
|
2001-06-15 05:28:30 +02:00
|
|
|
{
|
2003-02-10 00:33:26 +01:00
|
|
|
// !attention! "i" may not correspond to "this"
|
2001-10-31 05:27:46 +01:00
|
|
|
functor_t ifu = functor_index(i);
|
|
|
|
|
ifu->put(i, val);
|
|
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
if (i < ix+naddr)
|
|
|
|
|
{
|
|
|
|
|
if (update_addr_bit(this, i))
|
2001-10-14 03:36:12 +02:00
|
|
|
update_data(this);
|
2001-06-15 05:28:30 +02:00
|
|
|
}
|
|
|
|
|
|
2003-02-10 00:33:26 +01:00
|
|
|
// port ix+naddr is the write clock. If its input value is
|
|
|
|
|
// undefined, we do asynchronous write. Else any event on ix+naddr
|
2001-10-31 05:27:46 +01:00
|
|
|
// is a valid write clock edge. Connect an appropriate edge event
|
|
|
|
|
// functor.
|
2004-10-04 03:10:51 +02:00
|
|
|
|
2001-06-15 05:28:30 +02:00
|
|
|
if (i == ix+naddr
|
|
|
|
|
|| (writable && functor_get_input(ix+naddr) == 3))
|
|
|
|
|
{
|
|
|
|
|
assert(writable);
|
|
|
|
|
write_event(this);
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-06-15 05:28:30 +02:00
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
|
|
|
|
|
// %set/mem
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
void memory_set(vvp_memory_t mem, unsigned idx, unsigned char val)
|
|
|
|
|
{
|
2001-09-29 22:55:42 +02:00
|
|
|
if (idx/4 >= (mem->size * mem->fwidth))
|
2001-09-29 06:45:20 +02:00
|
|
|
return;
|
|
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
if (!set_bit(mem->bits, idx, val))
|
|
|
|
|
return;
|
|
|
|
|
|
2001-10-14 03:36:12 +02:00
|
|
|
unsigned widx = idx/(4*mem->fwidth);
|
|
|
|
|
unsigned bidx = idx%(4*mem->fwidth);
|
|
|
|
|
|
|
|
|
|
update_data_ports(mem, get_word_ix(mem, widx), bidx, val);
|
2001-05-01 03:09:39 +02:00
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
|
|
|
|
|
// %load/mem
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
unsigned memory_get(vvp_memory_t mem, unsigned idx)
|
|
|
|
|
{
|
2001-09-29 22:55:42 +02:00
|
|
|
if (idx/4 >= (mem->size * mem->fwidth))
|
2001-09-29 06:45:20 +02:00
|
|
|
return 2;
|
|
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
return get_bit(mem->bits, idx);
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2001-05-01 03:09:39 +02:00
|
|
|
// %assign/mem event scheduling
|
|
|
|
|
|
|
|
|
|
struct mem_assign_s: public vvp_gen_event_s
|
|
|
|
|
{
|
2004-10-04 03:10:51 +02:00
|
|
|
union
|
2001-05-01 03:09:39 +02:00
|
|
|
{
|
|
|
|
|
vvp_memory_t mem;
|
|
|
|
|
struct mem_assign_s *next;
|
|
|
|
|
};
|
|
|
|
|
unsigned long idx;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct mem_assign_s* ma_free_list = 0;
|
|
|
|
|
|
|
|
|
|
inline static struct mem_assign_s* ma_alloc()
|
|
|
|
|
{
|
|
|
|
|
struct mem_assign_s* cur = ma_free_list;
|
|
|
|
|
if (!cur)
|
|
|
|
|
cur = (struct mem_assign_s*) malloc(sizeof(struct mem_assign_s));
|
|
|
|
|
else
|
|
|
|
|
ma_free_list = cur->next;
|
2001-07-11 04:27:21 +02:00
|
|
|
|
2001-05-01 03:09:39 +02:00
|
|
|
return cur;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline static void ma_free(struct mem_assign_s* cur)
|
|
|
|
|
{
|
|
|
|
|
cur->next = ma_free_list;
|
|
|
|
|
ma_free_list = cur;
|
|
|
|
|
}
|
|
|
|
|
|
2005-03-03 05:33:10 +01:00
|
|
|
#if 0
|
2001-05-01 03:09:39 +02:00
|
|
|
static void run_mem_assign(vvp_gen_event_t obj, unsigned char val)
|
|
|
|
|
{
|
|
|
|
|
struct mem_assign_s *e = (struct mem_assign_s *) obj;
|
|
|
|
|
memory_set(e->mem, e->idx, val);
|
|
|
|
|
ma_free(e);
|
|
|
|
|
}
|
2005-03-03 05:33:10 +01:00
|
|
|
#endif
|
2002-08-12 01:47:04 +02:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* $Log: memory.cc,v $
|
2005-03-03 05:33:10 +01:00
|
|
|
* Revision 1.23 2005/03/03 04:33:10 steve
|
|
|
|
|
* Rearrange how memories are supported as vvp_vector4 arrays.
|
2002-08-12 01:47:04 +02:00
|
|
|
*
|
|
|
|
|
*/
|