/*
Copyright (C) 1998 T. Scott Dattalo
This file is part of gpsim.
gpsim is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
gpsim 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 gpasm; see the file COPYING. If not, write to
the Free Software Foundation, 59 Temple Place - Suite 330,
Boston, MA 02111-1307, USA. */
#include <stdio.h>
#include <iostream>
#include <iomanip>
#include "../config.h"
#include "12bit-processors.h"
#include <string>
#include "stimuli.h"
#include "trace.h"
extern unsigned int config_word;
//========================================================================
//
class OptionTraceObject : public RegisterWriteTraceObject
{
public:
OptionTraceObject(Processor *_cpu, OPTION_REG *pOptionReg, RegisterValue trv)
: RegisterWriteTraceObject(_cpu, pOptionReg, trv)
{
}
void print(FILE *fp)
{
char sFrom[16];
char sTo[16];
if(reg)
fprintf(fp, " Option: from 0x%s to 0x%s\n",
from.toString(sFrom,sizeof(sFrom)),
to.toString(sTo,sizeof(sTo)));
}
};
//========================================================================
class OptionTraceType : public TraceType
{
public:
OptionTraceType(Processor *_cpu, OPTION_REG *pOptionReg)
: TraceType(1), m_cpu(_cpu),m_pOptionReg(pOptionReg)
{
}
TraceObject *decode(unsigned int tbi)
{
unsigned int tv = trace.get(tbi);
RegisterValue rv = RegisterValue(tv&0xff,0);
OptionTraceObject *oto = new OptionTraceObject(m_cpu, m_pOptionReg, rv);
return oto;
}
int dump_raw(Trace *pTrace,
unsigned int tbi,
char *buf, int bufsize)
{
if (!pTrace)
return 0;
int n = TraceType::dump_raw(pTrace, tbi,buf,bufsize);
buf += n;
bufsize -= n;
unsigned int tv = pTrace->get(tbi);
//unsigned int subtype = (tv >> 8) & 0xfff;
int m = snprintf(buf, bufsize,
" Option Reg: was 0x%0X ", tv & 0xff);
return m>0 ? (m+n) : n;
}
protected:
Processor *m_cpu;
OPTION_REG *m_pOptionReg;
};
//-------------------------------------------------------------------
_12bit_processor::_12bit_processor(const char *_name, const char *desc)
: pic_processor(_name, desc)
{
pc = new Program_Counter();
pc->set_trace_command(trace.allocateTraceType(new PCTraceType(this,1)));
mOptionTT = new OptionTraceType(this,&option_reg);
trace.allocateTraceType(mOptionTT);
RegisterValue rv( (mOptionTT->type() & 0xff000000) | 0, 0);
option_reg.set_write_trace(rv);
option_reg.set_read_trace(rv);
}
_12bit_processor::~_12bit_processor()
{
}
void _12bit_processor::create_symbols()
{
pic_processor::create_symbols();
// add a special symbol for W
symbol_table.add_w(W);
}
void _12bit_processor::por()
{
pic_processor::por();
}
void _12bit_processor::reset(RESET_TYPE r)
{
option_reg.reset(r);
pic_processor::reset(r);
}
//-------------------------------------------------------------------
bool _12bit_processor::set_config_word(unsigned int address,unsigned int cfg_word)
{
// Clear all of the configuration bits in config_modes and then
// reset each of them based on the config bits in cfg_word:
//config_modes &= ~(CM_WDTE);
//config_modes |= ( (cfg_word & WDTE) ? CM_WDTE : 0);
//cout << " setting cfg_word and cfg_modes " << hex << config_word << " " << config_modes << '\n';
if((address == config_word_address()) && config_modes) {
config_modes->config_mode = (config_modes->config_mode & ~7) | (cfg_word & 7);
config_word = cfg_word;
if((bool)verbose && config_modes)
config_modes->print();
return true;
}
return false;
}
void _12bit_processor::create()
{
if(verbose)
cout << "_12bit_processor create, type = " << isa() << '\n';
pa_bits = 0; // Assume only one code page (page select bits in status)
pic_processor::create();
fsr = new FSR_12(fsr_register_page_bits(), fsr_valid_bits());
fsr->new_name("fsr");
// Sigh. Hack, hack,... manually assign indf bits
indf->fsr_mask = 0x1f;
indf->base_address_mask1 = 0x0;
indf->base_address_mask2 = 0x1f;
stack->stack_mask = 1; // The 12bit core only has 2 stack positions
//1 tmr0.set_cpu(this);
//1 tmr0.start(0);
}
//-------------------------------------------------------------------
void _12bit_processor::create_config_memory()
{
m_configMemory = new ConfigMemory *[1];
*m_configMemory = new ConfigMemory("CONFIG", 0,"Configuration Word",this,0xfff);
}
//-------------------------------------------------------------------
void _12bit_processor::option_new_bits_6_7(unsigned int bits)
{
//portb.rbpu_intedg_update(bits);
cout << "12bit, option bits 6 and/or 7 changed\n";
}
//-------------------------------------------------------------------
void _12bit_processor::dump_registers ()
{
pic_processor::dump_registers();
cout << "option = " << option_reg.value.get() << '\n';
}
syntax highlighted by Code2HTML, v. 0.9.1