/* 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 #include #include #include "../config.h" #include "12bit-processors.h" #include #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'; }