/* * Copyright (c) 2000 Stephen Williams (steve@icarus.com) * Copyright (c) 2001 Stephan Boettcher * * 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 */ #ifdef HAVE_CVS_IDENT #ident "$Id: memory.cc,v 1.19 2002/09/17 00:42:22 steve Exp $" #endif #include "memory.h" #include "symbols.h" #include "schedule.h" #include #ifdef HAVE_MALLOC_H #include #endif #include #include typedef struct vvp_memory_port_s *vvp_memory_port_t; struct vvp_memory_s { char *name; // VPI scope.name // Address port properties: unsigned size; // total number of data words unsigned a_idxs; // number of address indices vvp_memory_index_t a_idx; // vector of address indices // Data port properties: unsigned width; // number of data bits unsigned fwidth; // number of bytes (4bits) per data word int msb, lsb; // Most/Least Significant data bit (VPI) vvp_memory_bits_t bits; // Array of bits vvp_memory_port_t addr_root; // Port list root; }; unsigned memory_data_width(vvp_memory_t mem) { return mem->width; } #define VVP_MEMORY_NO_ADDR ((int)0x80000000) struct vvp_memory_index_s { int first; // first memory address unsigned size; // number of valid addresses // Added to correctly support vpiLeftRange and vpiRightRange int left; int right; }; struct vvp_memory_port_s : public functor_s { void set(vvp_ipoint_t i, bool push, unsigned val, unsigned str); vvp_memory_t mem; vvp_ipoint_t ix; unsigned naddr; vvp_memory_port_t next; int cur_addr; vvp_memory_bits_t cur_bits; unsigned bitoff; unsigned nbits; bool writable; }; unsigned memory_size(vvp_memory_t mem) { return mem->size; } unsigned memory_root(vvp_memory_t mem, unsigned ix) { if (ix >= mem->a_idxs) return 0; return mem->a_idx[ix].first; } unsigned memory_left_range(vvp_memory_t mem, unsigned ix) { if (ix >= mem->a_idxs) return 0; return mem->a_idx[ix].left; } unsigned memory_right_range(vvp_memory_t mem, unsigned ix) { if (ix >= mem->a_idxs) return 0; return mem->a_idx[ix].right; } unsigned memory_word_left_range(vvp_memory_t mem) { return mem->msb; } unsigned memory_word_right_range(vvp_memory_t mem) { return mem->lsb; } char *memory_name(vvp_memory_t mem) { return mem->name; } // Compilation static symbol_table_t memory_table = 0; vvp_memory_t memory_find(char *label) { if (memory_table == 0) return 0; symbol_value_t v = sym_get_value(memory_table, label); return (vvp_memory_t)v.ptr; } vvp_memory_t memory_create(char *label) { 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; } void memory_new(vvp_memory_t mem, char *name, int msb, int lsb, unsigned idxs, long *idx) { mem->width = msb > lsb ? msb-lsb+1 : lsb-msb+1; mem->msb = msb; mem->lsb = lsb; mem->fwidth = (mem->width+3)/4; assert((idxs&1) == 0); mem->a_idxs = idxs/2; mem->a_idx = (vvp_memory_index_t) malloc(mem->a_idxs*sizeof(struct vvp_memory_index_s)); assert(mem->a_idxs); mem->size = 1; for (unsigned i=0; i < mem->a_idxs; i++) { vvp_memory_index_t x = mem->a_idx + i; int msw = *(idx++); int lsw = *(idx++); x->left = msw; x->right = lsw; if (msw > lsw) { x->size = msw - lsw + 1; x->first = lsw; } else { x->size = lsw - msw + 1; x->first = msw; } mem->size *= x->size; } mem->bits = (vvp_memory_bits_t) malloc(mem->size * mem->fwidth); assert(mem->bits); memset(mem->bits, 0xaa, mem->size * mem->fwidth); mem->addr_root = 0x0; mem->name = name; } static void update_addr(vvp_memory_port_t addr); vvp_ipoint_t memory_port_new(vvp_memory_t mem, unsigned nbits, unsigned bitoff, unsigned naddr, bool writable) { unsigned nfun = naddr; if (writable) nfun += 2 + nbits; nfun = (nfun+3)/4; if (nfun < nbits) nfun = nbits; vvp_memory_port_t a = new struct vvp_memory_port_s; a->mem = mem; a->naddr = naddr; a->writable = writable; a->nbits = nbits; a->bitoff = bitoff; a->next = mem->addr_root; mem->addr_root = a; a->ix = functor_allocate(nfun); functor_define(a->ix, a); if (nfun > 1) { 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); } } a->cur_addr = VVP_MEMORY_NO_ADDR; a->cur_bits = 0x0; return a->ix; } void memory_init_nibble(vvp_memory_t mem, unsigned idx, unsigned char val) { assert(idx < mem->size*mem->fwidth); mem->bits[idx] = val; } // Utilities inline static vvp_memory_bits_t get_word_ix(vvp_memory_t mem, unsigned idx) { return mem->bits + idx*mem->fwidth; } inline static vvp_memory_bits_t get_word(vvp_memory_t mem, int addr) { assert(mem->a_idxs==1); unsigned waddr = addr - mem->a_idx[0].first; if (waddr >= mem->size) return 0x0; return get_word_ix(mem, waddr); } 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<> (2*(bit&3))) & 3; } inline static unsigned char functor_get_inputs(vvp_ipoint_t ip) { functor_t fp = functor_index(ip); assert(fp); return fp->ival; } inline static unsigned char functor_get_input(vvp_ipoint_t ip) { unsigned char bits = functor_get_inputs(ip); return (bits >> (2*ipoint_port(ip))) & 3; } static bool update_addr_bit(vvp_memory_port_t addr, vvp_ipoint_t ip) { unsigned abit = ip - addr->ix; assert(abit >= 0 && abit < addr->naddr); 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<cur_addr &=~ (1<cur_bits = get_word(addr->mem, addr->cur_addr); return addr->cur_addr != old; } static void update_addr(vvp_memory_port_t addr) { addr->cur_addr = 0; for (unsigned i=0; i < addr->naddr; i++) { update_addr_bit(addr, addr->ix+i); if (addr->cur_addr == VVP_MEMORY_NO_ADDR) break; } } inline static void update_data(vvp_memory_port_t data) { assert(data); for (unsigned i=0; i < data->nbits; i++) { vvp_ipoint_t dx = ipoint_index(data->ix, i); functor_t df = functor_index(dx); unsigned char out = get_bit(data->cur_bits, i + data->bitoff); df->put_oval(out, true); } } static void update_data_ports(vvp_memory_t mem, vvp_memory_bits_t bits, int bit, unsigned char val) { if (!bits) return; vvp_memory_port_t a = mem->addr_root; while (a) { if (bits == a->cur_bits) { unsigned i = bit - a->bitoff; if (i < a->nbits) { vvp_ipoint_t ix = ipoint_index(a->ix, i); functor_t df = functor_index(ix); df->put_oval(val, true); } } a = a->next; } } static inline void write_event(vvp_memory_port_t p) { if (!p->cur_bits) return; 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)) { // if a write would change the memory bit, but is // undefined (x or z), set the bit to x. if (we > 1) { set_bit(p->cur_bits, i + p->bitoff, 2); val = 2; } update_data_ports(p->mem, p->cur_bits, i + p->bitoff, val); } } } void vvp_memory_port_s::set(vvp_ipoint_t i, bool, unsigned val, unsigned) { // !attention! "i" may not corespont to "this" functor_t ifu = functor_index(i); ifu->put(i, val); if (i < ix+naddr) { if (update_addr_bit(this, i)) update_data(this); } // port ix+naddr is the write clock. If it's input value is // undefined, we do assynchronous write. Else any event on ix+naddr // is a valid write clock edge. Connect an appropriate edge event // functor. if (i == ix+naddr || (writable && functor_get_input(ix+naddr) == 3)) { assert(writable); write_event(this); } } // %set/mem void memory_set(vvp_memory_t mem, unsigned idx, unsigned char val) { if (idx/4 >= (mem->size * mem->fwidth)) return; if (!set_bit(mem->bits, idx, val)) return; unsigned widx = idx/(4*mem->fwidth); unsigned bidx = idx%(4*mem->fwidth); update_data_ports(mem, get_word_ix(mem, widx), bidx, val); } // %load/mem unsigned memory_get(vvp_memory_t mem, unsigned idx) { if (idx/4 >= (mem->size * mem->fwidth)) return 2; return get_bit(mem->bits, idx); } // %assign/mem event scheduling struct mem_assign_s: public vvp_gen_event_s { union { 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; cur->sync_flag = false; return cur; } inline static void ma_free(struct mem_assign_s* cur) { cur->next = ma_free_list; ma_free_list = cur; } 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); } void schedule_memory(vvp_memory_t mem, unsigned idx, unsigned char val, unsigned delay) { struct mem_assign_s *e = ma_alloc(); e->run = run_mem_assign; e->mem = mem; e->idx = idx; schedule_generic(e, val, delay); } /* * $Log: memory.cc,v $ * Revision 1.19 2002/09/17 00:42:22 steve * Proper initialization of the memories table. * * Revision 1.18 2002/08/12 01:35:08 steve * conditional ident string using autoconfig. * * Revision 1.17 2002/08/11 23:47:05 steve * Add missing Log and Ident strings. * */